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CAPÍTULO 2 


Certificação? 


As certificações J ava s ão, p elo b em ou p elo m al, muito r econhecidas no 
mercado. Em sua última versão, a principal certificação foi quebrada em 
duas provas. Este livro vai guiá-lo por questões e assuntos abordados para 
a primeira prova, a Java SE 7 Programmer I, de maneira profunda e 
desafiadora. 

O livro vai percorrer cada tema, com detalhes e exercícios, para você 
chegar à prova confiante. Decorar regras seria uma maneira de estudar, mas 
não estimulante. Por que não compila? Por que não executa como esperado? 
Mais do que um guia para que você tenha sucesso na prova, nossa intenção é 
mostrar como a linguagem funciona por trás. 

Ao terminar essa longa caminhada, você será capaz de entender melhor 
a linguagem, assim como poder dizer com exatidão os motivos de determi- 
nadas construções e idiomismos. 
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Como estudar 


Lembre-se de usar a linha de comando do Java, não use o Eclipse ou qual- 
quer outra IDE: os erros que o compilador da linha de comando mostra po- 
dem ser diferentes do da IDE, e você não quer que isso atrapalhe seu desem- 
penho. 

Lembre-se de ficar atento, na prova não ficará claro qual o assunto que 
está sendo testado e você deve se concentrar em todo o código, não só em um 
assunto ou outro. 

Esse processo é longo e a recomendação é que agende a prova agora 
mesmo no site da Oracle, para que não haja pausa desde o primeiro dia de 
leitura, até o último dia de leitura, a execução de diversos simulados e a prova 
em si. 

Não deixe de testar todo o código em que não sentir confiança. Os exer- 
cícios são gerados de propósito para causar insegurança no candidato, para 
levá-lo para um lado, sendo que o problema pode estar em outro. E faça 
muitos exercícios e simulados. 

Não hesite, tire suas dúvidas no site do GUJ e nos avise de sua certificação 
via twitter ou facebook: 

http://www.guj.com.br http://www-twitter.com/casadocodigo 
http://www.facebook.com/casadocodigo 

Bom estudo, boa prova, boa sorte e, acima de tudo, bem-vindo ao grupo 


daqueles que não só usam uma linguagem, mas a dominam. 


Seções da prova 


Os assuntos cobrados e abordados aqui são: 
1) Java Basics 


e Define the scope of variables 


Define the structure of a Java class 
e Create executable Java applications with a main method 


* Import other Java packages to make them accessible in your code 
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2) Working With Java Data Types 


e Declare and initialize variables 


e Differentiate between object reference variables and primitive vari- 
ables 


e Read or write to object fields 


e Explain an Objects Lifecycle (creation, “dereference” and garbage 
collection) 


e Call methods on objects 
e Manipulate data using the StringBuilder class and its methods 


e Creating and manipulating Strings 
3) Using Operators and Decision Constructs 


e Use Java operators 


e Use parenthesis to override operator precedence 


Test equality between Strings and other objects using == and equals 
0 


e Create if and if/else constructs 


Use a switch statement 


4) Creating and Using Arrays 


e Declare, instantiate, initialize and use a one-dimensional array 
e Declare, instantiate, initialize and use multi-dimensional array 


e Declare and use an ArrayList 
5) Using Loop Constructs 


e Create and use while loops 
e Create and use for loops including the enhanced for loop 


e Create and use do/while loops 
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e Compare loop constructs 


e Use break and continue 
6) Working with Methods and Encapsulation 


e Create methods with arguments and return values 

e Apply the static key word to methods and fields 

e Create an overloaded method 

e Differentiate between default and user defined constructors 
e Create and overload constructors 

e Apply access modifiers 

e Apply encapsulation principles to a class 


e Determine the effect upon object references and primitive values 
when they are passed into methods that change the values 


7) Working with Inheritance 


* Implement inheritance 


Develop code that demonstrates the use of polymorphism 


Differentiate between the type of a reference and the type of an object 
e Determine when casting is necessary 
e Use super and this to access objects and constructors 


e Use abstract classes and interfaces 
8) Handling Exceptions 


e Differentiate among checked exceptions, RuntimeExceptions and 
Errors 


e Create a try-catch block and determine how exceptions alter normal 
program flow 


e Describe what Exceptions are used for in Java 
e Invoke a method that throws an exception 


* Recognize common exception classes and categories 


CAPÍTULO 3 


O básico de Java 


3.1 DEFINA O ESCOPO DE VARIÁVEIS 


O escopo é o que determina em que pontos do código uma variável pode ser 
usada. 


Variáveis locais 


Chamamos de locais as variáveis declaradas dentro de métodos ou con- 
strutores. Antes de continuar, vamos estabelecer uma regra básica: o ciclo de 
vida de uma variável local vai do ponto onde ela foi declarada até o fim do 
bloco onde ela foi declarada. 

Mas o que é um bloco? Podemos entender como bloco um trecho de 
código entre chaves. Pode ser um método, um construtor, o corpo de um if, 
de um for etc.: 
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public void mi() { // início do bloco do método 
int x = 10; // variável local do método 


if (x >= 10) { // início do bloco do if 
int y = 50; // variável local do if 
System.out.print(y); 


} // fim do bloco do if 
} // fim do bloco do método 


Analisando esse código, temos uma variável x, que é declarada no 
começo do método. Ela pode ser utilizada durante todo o corpo do método. 
Dentro do if, declaramos a variável y. y só pode ser utilizada dentro do 
corpo do if, delimitado pelas chaves. Se tentarmos usar y fora do corpo do 
i £, teremos um erro de compilação, pois a variável saiu do seu escopo. 

Tome cuidado especial com loops for. As variáveis declaradas na área 
de inicialização do loop só podem ser usadas no corpo do loop: 


for (int i = 0, j = 0; i < 10; i++) 
jtt; 


System.out.println(j); // erro, já não está mais no escopo 


Parâmetros de métodos também podem ser considerados variáveis lo- 
cais ao método, ou seja, só podem ser usados dentro do método onde foram 
declarados: 


class Teste { 


public void mi(String bla) { 
System.out.printí(bla); 


public void m2() { 
// erro de compilação pois bla não existe neste 
// escopo 
System.out.println(bla); 
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Variáveis de instância 


Variáveis de instância ou variáveis de objeto são os atributos dos objetos. 
São declaradas dentro da classe, mas fora de qualquer método ou constru- 
tor. Podem ser acessadas por qualquer membro da classe e ficam em escopo 
enquanto o objeto existir: 


class Pessoa { 
// variável de instância ou variável de objeto 
String nome; 


public void setNome(String n) { 
// acessando a variável de instância no método 
this.nome = n; 


Variáveis estáticas (class variables) 


Podemos declarar variáveis que são compartilhadas por todas as instân- 
cias de uma classe usando a palavra chave static. Essas variáveis estão no 
escopo da classe, e lá ficarão enquanto a classe estiver carregada na memória 
(enquanto o programa estiver rodando, na grande maioria dos casos). 


class Pessoa { 
static int id = 1; 


class Teste { 
public static void main(Stringl[] args) { 
Pessoa p = new Pessoa(); 
System.out.println(p.id); // acessando pelo objeto 
System.out.println(Pessoa.id); // acessando direto pela 
// classe 
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No caso de variáveis static, não precisamos ter uma referência para 
usá-las e podemos acessá-las diretamente a partir da classe, desde que respei- 
tando as regras de visibilidade da variável. 


Variáveis com o mesmo nome 


Logicamente, não é possível declarar duas variáveis no mesmo escopo 
com o mesmo nome: 


public void bla() { 
int a = 0; 
int a = 10;// erro 


Mas, eventualmente, podemos ter variáveis em escopos diferentes que po- 
dem ser declaradas com o mesmo nome. Em casos em que possa haver am- 
biguidade na hora de declará-las, o próprio compilador irá emitir um erro 
evitando a confusão. Por exemplo, não podemos declarar variáveis de classe 
e de instância com o mesmo nome: 


class Bla { 
static int a; 
int a; // erro de compilação, 


System.out.println(new Bla().a); // qual variável estamos 
// acessando? 


Também não podemos declarar variáveis locais com o mesmo nome de 
parâmetros: 


public void metodo(String par) { 
int par = 0; // erro de compilação 


System.out.printin(par); // qual? 


Apesar de parecer estranho, é permitido declarar variáveis locais ou 
parâmetros com o mesmo nome de variáveis de instância ou de classe. Essa 
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técnica é chamada de shadowing. Nesses casos, é possível resolver a ambigu- 
idade: para variáveis de classe, podemos referenciar pela própria classe; para 
variáveis de instância, usamos a palavra chave this: 


class Pessoa { 


static int x = 0; 
int y = 0; 


public static void setX(int x) 1 
// Usando a referência da classe 


Pessoa.x = X; 


public void setY(int y) { 
// usando o this 
this.y = y; 


Quando não usamos o this ou o nome da classe para usar a variável, o 
compilador sempre utilizará a variável de menor escopo: 


class X { 
int a = 10; 


public void metodo() { 


int a = 20; // shadowing 
System.out.printin(a); // imprime 20 


1) Escolha a opção adequada ao tentar compilar e rodar o código a seguir: 


ı class Teste { 


2 public static void main(String[] args) { 
3 for (int i = 0; i < 20; i++) { 
4 System.out.println(i); 
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6 int i = 15; 

7 System.out.println(i); 
8 } 

9 } 


a) Erro de compilação na linha 6. A variável i não pode ser redeclarada. 
b) Erro de compilação na linha 7. A variável i é ambígua. 
c) Compila e roda, imprimindo de o até 19 e depois 15. 


d) Compila e roda, imprimindo de o até 19, depois ocorre um erro de 
execução na linha 6. 


e) Compila e roda, imprimindo de o até 19 e depois 19 novamente. 


2) Escolha a opção adequada ao tentar compilar e rodar o código a seguir: 


ı class Teste { 


2 static int x = 15; 

3 

4 public static void main(String[] x) 1 
5 x = 200; 

6 System.out.printlin(x); 

7 } 

aJ 


a) O código compila e roda, imprimindo 200. 
b) O código compila e roda, imprimindo 15. 
c) O código não compila. 


d) O código compila mas dá erro em execução. 


3) Escolha a opção adequada ao tentar compilar e rodar o código a seguir: 


ı class Teste { 


2 static int i = 3; 

3 

4 public static void main(Stringl] a) 1 

5 for (new Teste().i = 10; new Teste().i < 100; 
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6 new Teste().i++) { 

7 System. out .println(i); 
8 } 

9 } 

10 + 


a) Não compila a linha 4. 
b) Não compila a linha 5. 
c) Compila e imprime 100 vezes o número 3. 


d) Compila e imprime os números de 10 até 99. 


3.2 DEFINA A ESTRUTURA DE UMA CLASSE JAVA 


Nesta seção, iremos entender a estrutura de um arquivo java, onde inserir as 
declarações de pacotes e imports e como declarar classes e interfaces. 
Para entender a estrutura de uma classe, vamos ver o arquivo 


Pessoa. java: 


// Declaração de pacote 
package br.com.caelum.certificacao; 


// imports 
import java.util.Date; 


// Declaração da classe 
class Pessoa { 
// conteúdo da classe 


Pacotes 


Pacotes servem para separar e organizar as diversas classes que temos em 
nossos sistemas. Todas as classes pertencem a um pacote, sendo que, caso o 
pacote não seja explicitamente declarado, a classe fará parte do que chamamos 
de pacote padrão, ou default package. Todas as classes no default package 
se enxergam e podem ser utilizadas entre si. Classes no pacote default não 
podem ser importadas para uso em outros pacotes: 
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// Uma classe no pacote padrão 
classe Pessoa { 


AEF 


Para definir qual o pacote a que a classe pertence, usamos a palavra-chave 
package, seguida do nome do pacote. Só pode existir um único package 
definido por arquivo, e ele deve ser a primeira instrução do arquivo. Após 
a definição do package, devemos finalizar a instrução com um ;. Podem 
existir comentários antes da definição de um pacote: 


// declaração do pacote 
package br.com.caelum.certificacao; 


classe Pessoa { 


ll esi 


Aproveitando que tocamos no assunto, o package deve ser a primeira 
instrução de código que temos declarada em nosso arquivo. Comentários não 
são considerados parte do código, portanto, podem existir em qualquer lugar 
do arquivo java sem restrições. 


Para inserir comentário em nosso código, temos as seguintes formas: 


// comentário de linha 


/* 
comentário de 
multiplas linhas 
*/ 


class /* comentário no meio da linha */ Pessoa { 


[x 
* JavaDoc, repare que a primeira linha do comentário tem 
* 2 asteriscos 
*/ 

public void metodo() { 

F 
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PARA SABER MAIS: JAVADOC 


Javadoc é um tipo especial de comentário que pode ser utilizado 
para gerar uma documentação HTML a partir de nosso código. 
Para saber mais, acesse http://www.oracle.com/technetwork/java/javase/ 
documentation/javadoc-137458.html 





Classe 
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Uma classe é a forma no Java onde definimos os atributos e comporta- 


mentos de um objeto. A declaração de uma classe pode ser bem simples, ape- 


nas a palavra class seguida do nome e de (): 


ı class Pessoa {} 


Existem outros modificadores que podem ser usados na definição de uma 


classe, mas veremos essas outras opções mais à frente, onde discutiremos esses 


modificadores com mais detalhes. 


Vale lembrar que java é case sensitive e Class é o nome de uma classe e 


não podemos usá-lo para definir uma nova classe. 


Dentro de uma classe, podemos ter variáveis, métodos e construtores. Es- 


sas estruturas são chamadas de membros da classe.: 


ı class Pessoa { 


2 


String nome; 
String sobrenome; 


Pessoa(String nome, String sobrenome) { 


this.nome = nome; 
this.sobrenome = sobrenome; 


public String getNomeCompleto() 1 
return this.nome + this.sobrenome; 
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NOMES DOS MEMBROS 


Podemos ter membros de tipos diferentes com o mesmo nome. Fique 
atento, o código a seguir compila normalmente: 


| class B { 
4 String b; 
4 BO | 

} 


String b() { 
return null; 











Variaveis 


Usando como exemplo a classe Pessoa definida anteriormente, nome e 
sobrenome são variáveis. A declaração de variáveis é bem simples, sempre o 
tipo seguido do nome da variável. 

Dizemos que essas são variáveis de instância, pois existe uma cópia delas 
para cada objeto Pessoa criado em nosso programa. Cada cópia guarda o 
estado de uma certa instância desses objetos. 

Existem ainda variáveis que não guardam valores ou referências para uma 
determinada instância, mas sim um valor compartilhado por todas as instân- 
cias de objetos. Essas são variáveis estáticas, definidas com a palavra-chave 
static. Veremos mais sobre esse tipo de membro mais à frente. 


Métodos 


A declaração de métodos é um pouquinho diferente pois precisamos do 
tipo do retorno, seguido do nome do método e seguido de parênteses, sendo 
que pode ou não haver parâmetros de entrada desse método. Cada parâmetro 
é uma declaração de variável em si. Essa linha do método, onde está definido 
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o retorno, o nome e os parâmetros é onde temos a assinatura do método. 
Cuidado, pois a assinatura de um método inclui somente o nome do método 
e os tipos dos parâmetros. 

Assim como variáveis, métodos também podem ser static, como ver- 


emos mais adiante. 


Construtores 


Uma classe pode possuir zero ou vários construtores. Nossa classe 
Pessoa possui um construtor que recebe como parâmetros o nome e o so- 
brenome da pessoa. A principal diferença entre a declaração de um método 
e um construtor é que um construtor não tem retorno e possui o mesmo 
nome da classe. 





MÉTODOS COM O MESMO NOME DA CLASSE 


Cuidados com métodos que parecem construtores: 


class Executa { 


// construtor 
J Executa() { 
} 


4 // método 
void Executa() { 
| } 











Note que um construtor pode ter um return vazio: 


class X { 
int j = -100; 


X(int i) { 


17 


3.2. Defina a estrutura de uma classe Java Casa do Código 





5 if (i > 1) 
6 return; 
7 j= i; 

8 } 

ə > 


Caso o valor seja menor ou iguala 1, o valor de j será -100, caso 
contrário, será o mesmo valor de i. 


Interfaces 


Além de classes, também podemos declarar interfaces em nossos arquivos 
java. Para definir uma interface usamos a palavra reservada interface: 


ı interface Autenticavel { 


3 final int TAMANHO_SENHA = 8; 

4 

5 void autentica(String login, String senha); 
6} 


Em uma interface, devemos apenas definir a assinatura do método, sem 
a sua implementação. Além da assinatura de métodos, também é possível 
declarar constantes em interfaces. 


Multíplas estruturas em um arquivo 


Em java, é possível definir mais de uma classe/interface em um mesmo 
arquivo java, embora devamos seguir algumas regras: 


e Podem ser definidos em qualquer ordem; 


e Se existir alguma classe/interface pública, o nome do arquivo deve ser 
o mesmo dessa classe/interface; 


e Só pode existir uma classe/interface pública por arquivo; 


* Se não houver nenhuma classe/interface pública, o arquivo pode ter 
qualquer nome. 
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Logo, são válidos: 


// arquivol .java 
interface Bar (J 


class Foo {} 


// Foo. java 
public class Foo {} 


interface X {} 





PACOTES E IMPORTS EM ARQUIVOS COM MÚLTIPLAS ESTRU- 
TURAS 


As regras de pacotes e imports valem também para arquivos com 
múltiplas estruturas definidas. Caso exista a definição de um pacote, ela 
vale para todas as classes/interfaces definidas nesse arquivo, e o mesmo 


vale para imports. 











1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir 


sem nenhum parâmetro na linha de comando, como java D: 


ı package a.b.c; 
2 
3 import java.util.x; 


4 


s class D { 

6 public static void main(String[] args) { 

7 ArrayList<String> lista = new ArrayList<String>(); 
8 

9 for (String arg : args) { 

10 if (new E() .existe(arg)) 

u lista.add(arg); 

12 J 

13 F 
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1 import java.io.x; 


8 class E { 

19 public boolean existe(String nome) { 
20 File f = new File(nome); 

a return f.exists(); 

22 } 

3 + 


a) O arquivo não compila. 
b) O arquivo compila mas dá erro de execução pois o array é nulo. 


c) O arquivo compila mas dá erro de execução pois o array tem tamanho 


zero. 
d) Roda e imprime false. 
e) Roda e imprime true. 


f) Roda e não imprime nada. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class Teste { 


2 int Teste = 305; 

3 

4 void Teste() { 

5 System.out.printin(Teste); 

6 } 

7 

8 public static void main(Stringl] args) 1 
9 new Teste(); 

10 } 

nu } 


a) O código não compila: erros nas linhas 24, 25 e 26. 
b) O código não compila: erro na linha 25. 


c) O código não compila: erros nas linhas 24 e 26. 
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3) 


w 


4) 


w 


d) O código compila e, ao rodar, imprime 305. 
e) O código compila e não imprime nada. 


f) O código compila e, ao rodar, imprime uma linha em branco. 


Escolha a opção adequada ao tentar compilar o arquivo a seguir: 
package br.com.teste; 
import java.util.ArrayList; 


a) Erro na linha ı: definimos o pacote mas nenhum tipo. 
b) Erro na linha 3: importamos algo desnecessário ao arquivo. 


c) Compila sem erros. 


Escolha a opção adequada ao tentar compilar o arquivo A.java: 


class A implements B { 
} 
public interface B { 


E; 

class C extends A 1 

+ 

class D extends A, implements B { 


} 


a) Não compila: erro na linha 7. 
b) Não compila: erro na linha. 
c) Não compila: erro na linha 1, 5 e 7. 
d) Não compila: erro na linha 3. 


e) Compila. 
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3.3 CRIE APLICAÇÕES JAVA EXECUTÁVEIS COM UM 
MÉTODO MAIN 


Nesta seção, entenderemos as diferenças entre classes normais e classes que 
podem ser executadas pela linha de comando. 

Uma classe executável é uma classe que possui um método inicial para a 
execução do programa o método main, que será chamado pela JVM. Classes 
sem o método main não são classes executáveis e não podem ser usadas como 
ponto inicial da aplicação. 


Método main 


O tal método de entrada deve seguir algumas regras para ser executado 
pela JVM: 


e Ser público (public); 


Ser estático ( static); 
e Não ter retorno ( void); 
e Ter o nome main; 


e Receber como parâmetro um array ou varargs de String ( 


String[] ou String...). 


São então métodos main válidos os seguintes exemplos: 


//Parâmetro como array 
public static void main (Stringl] args) {} 


//Parâmetro como varargs 
public static void main (String... args) 1) 


//A ordem dos modificadores não importa 
static public void main(String[] args) 1) 


//O nome do parâmetro não importa 
P p 
public static void main (String... argumentos) () 
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3 //Também é uma definição válida de array 
4 public static void main (String args[]) 1) 


Executando uma classe pela linha de comando 


Para executar uma classe com main pela linha de comando, devemos 
compilar o arquivo com o comando javac e executar a classe com o co- 
mando java: 


Usando o arquivo HelloWorld. java a seguir: 


public class HelloWorld { 


3 public static void main(Stringl[] args) 1 
4 System.out.printin("Hello World! "); 
5 } 

6} 


Compilamos e executamos no terminal com os seguintes comandos: 


$ javac HelloWorld. java 


$ 
$ java HelloWorld 
Hello World! 


Repare que, para compilar a classe, passamos como parâmetro para o co- 
mando javac o nome do arquivo, enquanto para executar, passamos apenas 
o nome da classe ( HelloWorld) para o comando java. 


Passando parâmetros pelo linha de comando 


Ao executarmos uma classe pela linha de comando, podemos passar 
parâmetros para o método main. Esses valores serão recebidos no ar- 
ray do método main. Por exemplo, vamos passar um nome para a classe 
HelloWorld: 


ı public class HelloWorld{ 


3 public static void main(String[] args) { 
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//Lendo o valor da primeira posição do array args 
System.out.println("Hello " + args[0] + "!"); 


Para informar o valor do parâmetro, é só informá-lo APÓS o nome da 
classe que está sendo executada: 


java HelloWorld Mario 
Hello Mario! 


Você pode passar quantos parâmetros quiser, basta separá-los por espaço. 
Cada parâmetro informado será armazenado em uma posição do array, na 
mesma ordem em que foi informado. 


Compilação e execução 


Para criar um programa java, é preciso escrever um código-fonte e, através 
de um compilador, gerar o executável (bytecode). O compilador do JDK (Java 
Development Kit) é o javac. Para a prova de certificação, devemos conhecer o 
comportamento desse compilador. 

A execução do bytecode é feita pela JVM (Java Virtual Machine). O co- 
mando java invoca a máquina virtual para executar um programa java. Ao 
baixarmos o Java, podemos escolher baixar o JDK, que já vem com o JRE, ou 
somente o JRE (Java Runtime Environment), que inclui a Virtual Machine. 

Algumas questões da prova abordam aspectos fundamentais do processo 
de compilação e de execução. É necessário saber como os comandos javac 


eo java procuram os arquivos. 


javac 


Imagine o arquivo Prova. java dentro do diretório de meu projeto: 


class Prova { 
double tempo; 
F 


$ javac Prova.java 


24 


Casa do Código Capítulo 3. O básico de Java 





O bytecode da classe Prova gerado na compilação é colocado no ar- 
quivo Prova.class dentro do nosso diretório de trabalho, no meu caso, 
projeto. O resultado é: 


+ eto » | Prova.class 
E Prova.java 





Os projetos profissionais utilizam o recurso de pacotes para melhor orga- 
nizar os fontes e os bytecodes. Vejamos qual é o comportamento do javac 
com a utilização de pacotes. Colocamos o arquivo Prova. java no diretório 


certificacao: 


package certificacao; 
class Prova 1 

double tempo; 
} 


[certificacao]$ javac certificacao/Prova. java 


Nesse exemplo, o arquivo Prova.class é colocado no diretório 


certificacao. 
C projeto r centificacao » | Prova.class 





= Prova.java 





ESCOLHENDO A VERSÃO DO JAVA NA HORA DE COMPILAR 


Na hora da compilação, é possível definir em que versão do Java o 
código-fonte foi escrito. Isso é feito com a opção -source do comando 


javac. ( javac MinhaClasse. java -source 1.3). 
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java 


Vamos utilizar um exemplo para mostrar o funcionamento do comando 
java, criando o arquivo Teste. java no mesmo diretório, no mesmo pa- 
cote: 


package certificacao; 
class Teste { 
public static void main(Stringl] args) 1 
Prova p = new Prova(); 
210; 
System.out.printin(p.tempo); 


p. tempo 


} 


$ javac certificacao/Teste.java 
$ java certificacao.Teste 


Saída: 


210.0 


E o resultado são os arquivos: 





Somente o arquivo Teste. java foi passado para o compilador. Nesse 
arquivo, a classe Teste utiliza a classe Prova que se encontra em outro 
arquivo, Prova. java. Dessa forma, o compilador vai compilar automatica- 
mente o arquivo Prova. java se necessário. 

Para executar, é preciso passar o nome completo da classe desejada para 
a máquina virtual. O sufixo .class não faz parte do nome da classe, então 
ele não aparece na invocação da máquina virtual pelo comando java. 
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Propriedades na linha de comando 


A prova ainda cobra conhecimentos sobre como executar um programa 
java passando parâmetros ou propriedades para a JVM e essas propriedades 
são identificadas pelo -D antes delas. Este -D não faz parte da chave. 


java -Dchavel=abc -Dchave2=def Foo xpto bar 


chavel=abc e chave2=def são parâmetros/propriedades e xpto e 
bar são argumentos recebidos pelo método main. 


Classpath 


Para compilar ou para executar, é necessário que os comandos javac e 
java possam encontrar as classes referenciadas pela aplicação java. 

A prova de certificação exige o conhecimento do algoritmo de busca das 
classes. As classes feitas pelo programador são encontradas através do class- 
path (caminho das classes). 

O classpath é formado por diretórios, jars e zips que contenham as 
classes e pacotes da nossa aplicação. Por padrão, o classpath está configurado 
para o diretório corrente ( .). 


Configurando o classpath 


Há duas maneiras de configurar o classpath: 

1) Configurando a variável de ambiente CLASSPATH no sistema opera- 
cional. 

Basta seguir as opções do SO em questão e definir a variável. Isso é con- 
siderado uma má prática no dia a dia porque é um classpath global, que vai 
valer para qualquer programa java executado na máquina. 

2) Com as opções -cp ou -classpath dos comandos javac ou 
java. 

É a forma mais usada. Imagine que queremos usar alguma biblioteca 
junto com nosso programa: 


$ javac -cp /diretorio/biblioteca.jar Prova.java 
$ java -cp /diretorio/biblioteca. jar Prova 
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E podemos passar tanto caminhos de outras pastas como de JARs 
ou zips. Para passar mais de uma coisa no classpath, usamos o 
separador de parâmetros no SO (no Windows é ponto e vírgula, no 
Linux/Mac/Solaris/Unix são dois pontos): 


$ javac -cp /diretorio/biblioteca.jar;/outrodir/ scjp/Prova.java 
$ java -cp /diretorio/biblioteca.jar;/outrodir/ scjp.Prova 





PARA SABER MAIS: ARQUIVOS JAR 


Para facilitar a distribuição de bibliotecas de classes ou de aplicativos, 
o JDK disponibiliza uma ferramenta para a compactação das classes java. 

Um arquivo JAR nada mais é que a pasta de nossas classes no formato 
ZIP mas com extensão .jar. 


Para criar um jar incluindo a pasta scjp que fizemos antes: 


jar -cf bib.jar scjp 


Agora podemos executar nossa classe usando esse jar: 


java -cp bib.jar scjp.Prova 
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PARA SABER MAIS: META-INE/MANIEEST.ME 


Ao criar o jar usando o comando jar do JDK, ele cria automatica- 
mente a pasta META-INF, que é usada para configurações relativas ao 





nosso jar. E dentro dela, cria o arquivo Manifest .mf. 

Esse arquivo pode ser usado para algumas configurações. Por exem- 
plo, é possível dizer qual classe do nosso jar é a classe principal (Main- 
Class) e que deve ser executada. 

Basta criar um arquivo chamado Manifest .mf com a seguinte in- 


strução indicando a classe com o método main: 
Main-Class: scjp.Teste 

E depois gerar o jar passando esse arquivo: 
jar -cfm bib.jar meumanifest scjp 

Na hora de rodar um jar com Main-Class, basta usar: 


java -jar bib.jar 











1) Qual é uma assinatura válida do método main para executar um pro- 


grama java? 





a) public static void main(String... args) 


b) public static int main(String[] args) 








c) public static Void main(String []args) 


d) protected static void main(String[] args) 

















e) public static void main(int argc; String[] args) 


2) Escolha a opção adequada para compilar e rodar o arquivo A. java, ex- 


istente no diretório b: 
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3) 


n 
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package b; 
class AÍ 
public static void main(Stringl[] args) 1 
System.out.println('"rodando'"); 


a) javac Ae java A 

b) javac A.javae java A 

c) javac b/A.javae java A 
d) javac b/A.javae java b.A 


e) javac b.A.javae java b.A 





f) javac b.Ae java b.A 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(Stringl] args) 1 
System.out.printin(args); 
System.out.printin(args.length); 
System.out.printin(args[0]); 


a) Não compila: array não possui membro length. 
b) Não compila: o método print In não consegue imprimir um array. 


c) Ao rodar sem argumentos, ocorre uma NullPointerException na 





linha 5. 











d) Ao rodar sem argumentos, ocorre uma NullPointerExceptionna 





linha 4. 


e) Ao rodar sem argumentos, são impressos os valores “1” e “A”. 


«o» 


f) Ao rodar com o argumento “certificacao”, são impressos os valores “2 
CA» 
e A, 
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4) 


5) 


Escolha a opção adequada para rodar a classe A. java presente no di- 
retório b, que foi compactado em um arquivo chamado programa. jar, 
sendo que não existe nenhum arquivo de manifesto: 


package b; 
class AÍ 
public static void main(Stringl] args) 1 
System.out.printin(args[0]); 


a) java jar programa. jar 

b) java jar programa.jar b.A 
c) java -jar programa. jar 

d) java -jar programa.jar b.A 


e) java -cp programa. jar 





f) java -cp programa. jar b.A 


Escolha a opção adequada para compilar a classe A. java, definida como 
no pacote b presente no diretório b, e adicionar também o arquivo 
programa. jar na busca de classes durante a compilação. Lembre-se que 
. significa o diretório atual. 


a) javac -cp b.A.java -cp programa. jar 
b) javac -jar programa.jar b.A. java 

c) javac -cp programa.jar:b A.java 

d) javac -cp programa.jar:. b.A. java 
e) javac -cp . -cp programa. jar 

f) javac -jar programa.jar:. b/A.java 


g) javac -cp programa.jar:b b/A. java 





h) javac -cp programa. jar:. b/A.java 
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3.4 IMPORTE OUTROS PACOTES JAVA E DEIXE-OS 
ACESSÍVEIS AO SEU CÓDIGO 


Se duas classes estão no mesmo pacote, elas se “enxergam” entre si, sem a 
necessidade de colocar o nome do pacote. Por exemplo, imagine que as classes 





Pessoa € Endereco estejam no mesmo pacote: 


package modelo; 


class Endereco { 
String rua; 
String numero; 
String bairro; 


Pb isa 


E o outro arquivo: 


package modelo; 


class Pessoa { 
Endereco endereco; // Pessoa usando o endereço 


Para usar uma classe que está em outro pacote, temos duas opções: pode- 
mos referenciá-la usando o que chamamos de Full Qualified Name, ou 
seja, o nome do pacote seguido do nome da classe. O código ficaria assim: 


package financeiro; 


class Pedido { 
modelo.Pessoa cliente; // Usando a classe Pessoa de outro 
// pacote 


Tentamos compilar mas ele não deixa, porque uma classe, por padrão, só 
pode ser acessada dentro do próprio pacote, e a nossa classe Pessoa está no 
pacote modelo. Portanto, definiremos nossa classe Pessoa como pública. 
Veremos com mais calma os modificadores de acesso na seção que cobra isso 


32 


N 


Casa do Código Capítulo 3. O básico de Java 





na prova. Por enquanto, basta lembrar que classes públicas podem ser aces- 
sadas por outros pacotes, já classes padrão não podem. 


package modelo; 


public class Pessoa { 
Endereco endereco // Pessoa usando o endereço 


Outra opção é importar a classe Produto e referenciá-la apenas pelo 
nome simples dentro de nosso código. Para fazer o import usamos a palavra 
import, seguida do Full Qualified Name da classe. A instrução de 
import deve aparecer na classe logo após o package (se este existir), e antes 
da definição da classe. É possível importar mais de uma classe por vez: 


package modelo; 


// Importando a classe Produto do pacote estoque 
import estoque.Produto; 

// Outro import qualquer 

import java.util.Date; 


class Pedido { 
Pessoa cliente; // mesmo pacote 
Produto item; // importado 
Date dataEmissao; //importado 


Também é possível importar todas as classes de um determinado pacote, 
basta usar um » após o nome do pacote: 


// Importando todas as classes do pacote estoque 
import estoque.*; 
Importando classes com mesmo nome 


Quando precisamos usar duas classes com o mesmo nome mas de pacotes 
diferentes, só podemos importar uma delas. A outra deve ser referenciada 
pelo Full Qualified Name. Tentativas de importar as duas classes irão 
resultar em erros de compilação: 
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import java.util.Date; 
import java.sql.Date; // Erro de compilação pois temos duas 
// classes Date 


class Teste { 
Date d1; 
Date d2; 


O correto seria: 


import java.util.Date; 


class Teste 1 
Date d1; // java.util 
java.sql.Date d2; // java.sql 


Caso tenhamos um import específico e um import genérico, o Java usa o 
específico: 


import java.util.*; 
import java.sql.Date; 


class Teste{ 
Date dl; // java.sql 
Date d2; // java.sql 


Por padrão, todas as classes do pacote java. lang são importadas. Um 
ponto importante é que nenhuma classe de pacote que não seja o padrão pode 
importar uma classe do pacote padrão: 


class Gerente { 


} 


package modelo; 
classe Banco { 
Gerente gerente; // não compila pois não é possível importar 
// tipos do pacote padrão de jeito *nenhum* 
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Pacotes 


Nesta seção, entenderemos mais a fundo como funciona a declaração de 
pacotes, e como isso influencia nos imports das classes. 

Como já discutimos anteriormente, pacotes servem para organizar suas 
classes e interfaces. Eles permitem agrupar componentes que tenham alguma 
relação entre si, além de garantir algum nível de controle de acesso a mem- 
bros. Além de serem uma divisão lógica para as suas classes, os pacotes tam- 
bém definem uma separação física entre os arquivos de seu projeto, já que 
espelham a estrutura de diretórios dos arquivos do projeto. 


Subpacotes e estrutura de diretórios 


Pacotes são usados pela JVM como uma maneira de encontrar as classes 
no sistema de arquivos, logo a estrutura de diretórios do projeto deve ser a 
mesma da estrutura de pacotes. Vamos usar como exemplo a classe Pessoa: 


package projeto.modelo; 
public class Pessoa (J 


O arquivo Pessoa.java deve estar localizado dentro do diretório 
modelo, que deve estar dentro do diretorio projeto, conforme a figura 


a seguir: 
modelo b- -a | 
C projeto ' CREDO Peo 


Dizemos que modelo é um subpacote de projeto, já que está 
dentro dele. Podemos ter vários subpacotes, como projeto.utils e 
projeto.conversores, por exemplo. Usamos o caractere . como sepa- 
rador de pacotes e subpacotes. 
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Convenções de nomes para pacotes 


Existem algumas convenções para nomes de pacotes. Elas não são obri- 


gatórias, mas geralmente são seguidas para facilitar o entendimento e organi- 


zação do código: 


e O nome do pacote deve ser todo em letras minúsculas; 
e Um pacote deve começar com o site da empresa, ao contrário; 
e Após o site, deve vir o projeto; 


e Após o projeto, a estrutura é livre. 


Import usando classes de outros pacotes 


Existem diversas maneiras de referenciar uma classe de pacote diferente 


em nosso código. Vamos analisar essas opções: 


Full Qualified Name 


Podemos referenciar uma classe em nosso código usando o que 


chamamos de Full Qualified Name, ou FQN. Ele é composto pelo pacote 
completo mais o nome da classe, por exemplo: 


class Pessoa { 


// FQN da classe Calendar 
java.util.Calendar dataDeNascimento; 


import 


Usar o FQN nem sempre deixa o código legível, portanto, em vez de usar o 


nome completo da classe, podemos importá-la e usar apenas o nome simples 


da classe: 


import java.util.Calendar; 


class Pessoa { 
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É permitido também importar todas as classes de um pacote de uma vez, 
usando o » no lugar do nome da classe: 


import java.util.x; 


class Pessoa 1 
// Calendar e List são do pacote java.util 
Calendar dataDeNascimento; 
List<String> apelidos; 


Caso existam duas classes com o mesmo nome, mas de pacotes diferentes, 
só podemos importar uma delas. A outra deve ser referenciada pelo FQN: 


import java.util.Date; 


class Foo { 
//do java.util 
Date some; 
java.sql.Date other; 





} 
] 
MULTIPLOS IMPORTS COM * 
Caso importemos dois ou mais pacotes que contenham classes com 
o mesmo nome, será obrigatório especificar, usando o FỌN, qual das 
classes queremos utilizar. Ao tentar usar apenas o nome simples da classe, 
teremos um erro de compilação: 
import java.util.x; 
4 import java.sql.*; 
4 public class Testes { 
$ private Date d; // Erro de compilação, de qual pacote é 
// para usar? 
+ 
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Import de subpacotes 


Em Java, não podemos importar todas as classes de subpacotes usando 
*. Veja a seguinte situação, considerando que cada classe foi definida em seu 
próprio arquivo: 


package sistema.prova; 


public class Pergunta {} 


package sistema.banco; 


public class PerguntaDao (J 


package sistema; 


public class Exame {} 


package sistema.teste; 
import sistema.+*; //só importou a classe Exame 


public class Teste {} 


O único modo de importar todas as classes é explicitamente importando 
cada subpacote: 


package sistema.teste; 


import sistema.*; 
import sistema.prova.X*; 
import sistema.banco.x*; 
public class Teste {} 


import static 


Desde o Java 5, é possível importar apenas métodos e atributos estáticos 
de uma classe, usando a palavra-chave static juntamente como import. 
Podemos importar um a um ou simplesmente importar todos usando + 
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ı package model; 
3 public class Utils { 


4 


5 // Atributo estático público 

6 public static int VALOR = 0; 

7 // Métodos estáticos públicos 

8 public static void metodo1() 4{} 

9 public static void metodol (int a) 1) 
un } 


ı // Importando todos os membros public static de Utils 
2 import static model.Utils.x*; 


3 
4 public class Testes { 


5 


6 public static void main(Stringl[] args) { 
7 int x = VALOR; 

8 metodol (); 

9 metodol (x); 

10 } 

un } 


1) Escolha a opção adequada ao tentar compilar e rodar o Teste. Arquivo 
no diretório atual: 


import modelo.Cliente; 


2 class Teste { 


3 public static void main(String[] args) { 
4 new Cliente("guilherme") .imprime(); 
5 } 

6} 


Arquivo no diretório modelo: 


package modelo; 


class Cliente { 


w 
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private String nome; 
Cliente(String nome) 1 
this.nome = nome; 


} 
public void imprime() { 
System. out .print1n (nome); 


a) Não compila: erro na classe Teste. 
b) Não compila: erro na classe Cliente. 
c) Erro de execução: método main. 


d) Roda e imprime “Guilherme”. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


import modelo.basico.Cliente; 
import modelo.avancado.Cliente; 


class Teste { 
public static void main(String[] args) { 
System. out .println("Bem vindo!"); 


a) O código não compila, erro ao tentar importar duas classes com o 


mesmo nome. 


b) O código compila, mas ao rodar dá erro por ter importado duas classes 


com o mesmo nome. 


c) O código compila e roda imprimindo Bem vindo !, uma vez que nen- 
huma das classes importadas é usada no código, não existe ambigu- 
idade. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir, 
sabendo que existem duas classes Cliente, uma no pacote basico e 


outra no pacote avancado: 


Casa do Código Capítulo 3. O básico de Java 





4) 


5) 


import modelo.basico.Cliente; 
import modelo.avancado.X*; 


class Teste { 
public static void main(Stringl] args) 1 
System.out.printin("Bem vindo!"); 


a) O código não compila, erro ao tentar importar duas classes com o 
mesmo nome. 


b) O código compila mas ao rodar dá erro por ter importado duas classes 
com o mesmo nome. 


c) O código compila e roda imprimindo Bem vindo!. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


import modelo.basico.Cliente; 
import modelo.basico.Cliente; 


class Teste 1 
public static void main(Stringl] args) 1 
System.out.printin("Bem vindo!"); 


a) O código não compila, erro ao tentar importar duas classes com o 
mesmo nome. 


b) O código compila, mas ao rodar dá erro por ter importado duas classes 
com o mesmo nome. 


c) O código compila e roda imprimindo Bem vindo!, uma vez que não 
há ambiguidade. 


Escolha a opção adequada ao tentar compilar os arquivos a seguir: 


a/A. java! 
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package a; 
class A { 
b.B variavel; 


a/C.java: 


package a; 
class Cf 
b.B variavel; 


a/b/B.java: 


package a.b; 
class B { 
} 


a) Erro de compilação somente no arquivo A. 


b) Erro de compilação somente no arquivo B. 


c) Erro de compilação somente no arquivo C. 


d) Erro de compilação nos arquivos A e B. 
e) Erro de compilação nos arquivos A e C. 
f) Erro de compilação nos arquivos B e C. 


g) Compila com sucesso. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


package A; 
class B{ 


public static void main(String[] a) { 


System.out.printin('rodei'); 


a) Não compila: a variável do método main deve se chamar args. 
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w 


8) 


b) Não compila: pacote com letra maiúscula. 
c) Compila mas não roda: a classe B não é pública. 


d) Compila e roda. 


Escolha a opção adequada ao tentar compilar os arquivos a seguir: 
a/A.java: 
package a; 
public class A { 
public static final int VALOR = 15; 


public void executa(int x) { 
System.out.printin(x); 


b/B. java: 


package b; 
import static a.A.x; 
class BÍ 
void m() 1 
A a = new A(); 
a.executa (VALOR) ; 


a) B não compila: erro na linha 2. 
b) B não compila: erro na linha 5. 
c) B não compila: erro na linha 6. 


d) Tudo compila. 


Escolha a opção adequada ao tentar compilar os arquivos a seguir: 


a/A. java: 
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package a; 
public class A { 
public static final int VALOR = 15; 
public void executa(int x) { 
System.out.printin(x); 


b/B. java: 


package b; 
import a.A; 
static import a.A.x; 
class B{ 
void m() { 
A a = new A(); 
a.executa (VALOR) ; 


a) B não compila: erro na linha 3. 
b) B não compila: erro na linha 5. 
c) B não compila: erro na linha 6. 


d) Tudo compila. 


Escolha a opção adequada ao tentar compilar os arquivos a seguir: 


A.java: 


public class A É 
public static final int VALOR = 15; 
public void executa(int x) { 
System.out.printin(x); 


b/B. java: 
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1 package b; 

2 import static A.x; 

3 class B{ 

4 void m() { 

5 A a = new A(); 

6 a.executa (VALOR) ; 
7 } 

8 } 


a) Não compila 


b) Tudo compila. 
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CAPÍTULO 4 


Trablhando com tipos de dados 
em Java 


4.1 DECLARAR E INICIALIZAR VARIÁVEIS 


Qualquer programa de computador precisa manter informações de alguma 
forma. As linguagens de programação permitem a criação de variáveis para 
que possamos armazenar informações. Por exemplo, se precisarmos guardar 
a idade de uma pessoa, podemos utilizar uma variável que seja capaz de man- 
ter números inteiros. 

Quando precisamos de uma nova variável, devemos declarar que quere- 
mos criá-la. A declaração de variável no Java, obrigatoriamente, deve infor- 
mar o tipo e o nome que desejamos para ela. Por isso, essa linguagem é dita 
explicitamente tipada (todas as variáveis precisam ter o seu tipo definido). 
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// Declaração de uma variável chamada idade do tipo primitivo int 
int idade; 


Nem toda linguagem exige que as variáveis sejam iniciadas antes de serem 
utilizadas. Mas, no Java, a inicialização é obrigatória e pode ser implícita ou 
explícita. É de fundamental importância saber que, para usar uma variável, é 
necessário que ela tenha sido iniciada explicitamente ou implicitamente em 
algum momento antes da sua utilização. 

Variáveis locais (declaradas dentro de métodos/construtores) devem ser 
iniciadas antes de serem utilizadas, ou teremos um erro de compilação: 


// Declaração 
int idade; 


System.out.println(idade); // erro de compilação 


// Declaração 
int idade; 


// Inicialização explícita de uma variável 
idade = 10; 


// Utilização da variável 
System.out .printin(idade); // ok 


Podemos declarar e iniciar a variável na mesma instrução: 


// Declaração e inicialização explícita na mesma linha 
double pi = 3.14; 


Se eu tenho um if, a inicialização deve ser feita em todos os caminhos 
possíveis: 


void metodo(int a) { 
double x; 
if(a > 1) { 
x = 6; 
} 
System.out.println(x); // talvez x não tenha sido 
// inicializado, portanto não compila 
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Quando a variável é membro de uma classe, ela é iniciada implicitamente 
junto com o objeto com um valor default: 


class Prova { 
double tempo; 


// Implicitamente, na criação de um objeto Prova, 
// o atributo tempo é iniciado com O 
Prova prova = new Prova(); 


// Utilização do atributo tempo 
System.out.printin(prova.tempo) ; 


Outro momento em que ocorre a inicialização implícita é na criação de 
arrays: 


int [] numeros = new int[10]; 


System.out.println (numeros[0]); // imprime O 


Quando iniciadas implicitamente, os valores default para as variáveis são: 


e primitivos numéricos inteiros o 

* primitivos numéricos com ponto flutuante 0.0 
e boolean false 

e char vazio, equivalente a o 


e referências null 


Os tipos das variáveis do Java podem ser classificados em duas categorias: 
primitivos e não primitivos (referências). 
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Tipos primitivos 


Todos os tipos primitivos do Java já estão definidos e não é possível criar 
novos tipos primitivos. São oito os tipos primitivos do Java: byte, short, 
char, int, long, float, double e boolean. 

O boolean é o único primitivo não numérico. Todos os demais ar- 
mazenam números: double e float são ponto flutuante, e os demais, to- 
dos inteiros (incluindo char). Apesar de representar um caractere, o tipo 
char armazena seu valor como um número positivo. Em Java, não é pos- 
sível declarar variáveis com ou sem sinal (unsigned), todos os números (ex- 
ceto char) podem ser positivos e negativos. 

Cada tipo primitivo abrange um conjunto de valores. Por exemplo, o tipo 
byte abrange os números inteiros de -128 até 127. Isso depende do tamanho 
em bytes do tipo sendo usado. 


Os tipos inteiros têm os seguintes tamanhos: 


byte 1 byte (8 bits, de -128 a 127); 

e short 2 bytes (16 bits, de -32.768 a 32.767); 

e char 2 bytes (só positivo), (16 bits, de o a 65.535); 

e int 4 bytes (32 bits, de -2.147.483.648 a 2.147.483.647); 


e long 8 bytes (64 bits, de -9.223.372.036.854.775.808 a 
9.223.372.036.854.775.807). 





DECORAR O TAMANHO DOS PRIMITIVOS PARA PROVA 


Não há a necessidade de decorar o intervalo e tamanho de todos os 
tipos de primitivos para a prova. O único intervalo cobrado é o do byte 
(-127 a 128). 

É importante também saber que o char, apesar de ter o mesmo 
tamanho de um short, não consegue armazenar todos os números que 
cabem em um short,já queo char só armazena números positivos. 
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PARA SABER MAIS: CALCULANDO O INTERVALO DE VALORES 
Dado o número de bits N do tipo primitivo inteiro, para saber os val- 
ores que ele aceita usamos a seguinte conta: 
-2"(n-1) a 2"(n-1) -1 
O char, por ser apenas positivo, tem intervalo: 


O a 27(16) -1 











Os tipos ponto flutuante têm os seguintes tamanhos em notação científica: 
e float 4 bytes (32 bits, de +/-1.4 * 10 ^45 a +/-3.4028235 * 10 ^38); 


e double 8 bytes (64 bits, de +/-4.9 * 10 1324 a +/-1.7976931348623157 
* 10 1308). 


Todos os números de ponto flutuante também podem assumir os 
seguintes valores: 


e +/- infinity 
e +/-0 


e NaN (Not a Number) 


Literais 


Na codificação, muitas vezes o programador coloca os valores das var- 
jáveis diretamente no código-fonte. Quando isso ocorre, dizemos que o valor 
foi literalmente escrito no código, ou seja, é um valor literal. 

Todos os valores primitivos maiores que int podem ser expressos literal- 
mente. Por outro lado, as referências (valores não primitivos) não podem ser 
expressas de maneira literal (não conseguimos colocar direto os endereços de 
memória dos objetos). 

Ao inicializar uma variável, podemos explicitar que queremos que ela seja 
do tipo double ou long usando a letra específica: 
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// compila pois 737821237891232 é um double válido 
System.out.println(737821237891232d); 


// compila pois 737821237891232 é um long válido 
System.out.println(7378212378912321); 


// nao compila pois 737821237891232 é um valor maior que 
// o int aceita 
System.out.println(737821237891232); 


Da mesma maneira, o compilador é um pouco esperto e percebe se você 
tenta quebrar o limite de um int muito facilmente: 


// compila pois 7378212378912321 é um long válido 
long 1 = 7378212378912321; 


// não compila pois o compilador não é bobo assim 
int i = 1; 


// booleanos 
System.out.printin(true); // booleano verdadeiro 
System.out.printin(false); // booleano falso 


// números simples são considerados inteiros 
System.out.println(1); // int 


// números com casa decimal são considerados double. 

// Também podemos colocar uma letra "D" ou "d" no final 
System.out.println(1.0); //double 
System.out.println(1.0D); //double 


// números inteiros com a letra "L" ou "1" 
// no final são considerados long. 
System.out.printin(1L); //long 


// números com casa decimal com a letra "F" ou "f" 


// no final são considerados float. 
System.out.println(1.0F); //float 
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Bases diferentes 


No caso dos números inteiros, podemos declarar usando bases diferentes. 
O Java suporta a base decimal e mais as bases octal, hexadecimal e binária. 

Um número na base octal tem que começar com um zero à esquerda e 
pode usar apenas os algarismos de o a 7: 


int i = 0761; // base octal 


System.out.printin(i); // saída: 497 


E na hexadecimal, começa com 0x ou 0X e usa os algarismos de o a 15. 
Como não existe um algarismo “15”, usamos letras para representar algaris- 
cc » « » CA» cC » paid º EA 
mos de “10” a “15”, no caso, “A” a “F”, maiúsculas ou minúsculas: 


int j = 0xAB3400; // base hexadecimal 
System.out.println(j); // saída: 11219968 


cm 


Já na base binária, começamos com 0b, e só podemos usar “o” e “1”: 


int b = 0b100001011; // base binária 
System.out.printin(b); // saída: 267 


Não é necessário aprender a fazer a conversão entre as diferentes bases 
e a decimal. Apenas saber quais são os valores possíveis em cada base, para 
identificar erros de compilação como o que segue: 


int i = 0769; // erro, base octal não pode usar 9 


Notação científica 
Ao declarar doubles ou floats, podemos usar a notação científica: 


double d = 3.1E2; 
System.out.printin(d); // 310.0 


float e = 2e3f; 
System.out.printin(e); // 2000.0 


float f = 1E4F; 
System.out.printin(f); // 10000.0 
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Usando underlines em literais 


A partir do Java 7, existe a possibilidade de usarmos underlines (. ) quando 
estamos declarando literais para facilitar a leitura do código: 


int a = 123 456 789; 


Existem algumas regras sobre onde esses underlines podem ser posiciona- 
dos nos literais, e caso sejam colocados em locais errados resultam em erros 
de compilação. A regra básica é que eles só podem ser posicionados com val- 
ores numéricos em ambos os lados. Vamos ver alguns exemplos: 


int vi = 0 100 267 760; // ok 
int v2 = 0 x 4 13; // erro, _ antes e depois do x 
int v3 = Ob x10 BA 75; // erro, _ depois do b 
int v4 = Ob 10000 10 11; // erro, _ depois do b 
int v5 = 0xal0O AF 75; // ok, apesar de ser letra 
// representa dígito 
int v6 = 123 341; // erro, inicia com _ 
int v7 = 123 432; // erro, termina com _ 
int v8 = 0x1 0A0 11; // ok 
int v9 = 144 21 12; // ok 


A mesma regra se aplica a números de ponto flutuante: 


double di = 345.45 e3; // erro, _ antes do e 
double d2 = 345.45e 3; // erro, _ depois do e 
double d3 = 345.4 5e3; // ok 


double d4 = 34 5.45e3 2; // ok 
double d5 = 3 4 5.4 5e3; // ok 


double d6 = 345. 45F; // erro, _ depois do 

double d7 = 345 .45; // erro, _ antes do . 

double d8 = 345.45 F; // erro, _ antes do indicador de 
// float 

double d9 = 345.45 d; // erro, _ antes do indicador de 
// double 
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Iniciando chars 


Os chars são iniciados colocando o caractere desejado entre aspas sim- 
ples: 


char c = 'A!; 


Mas podemos iniciar com números também. Neste caso, o número rep- 
resenta a posição do caractere na tabela unicode: 


char c = 65; 
System.out.printin(c); // imprime A 


Não é necessário decorar a tabela unicode, mas é preciso prestar atenção 
a pegadinhas como a seguinte: 


char sete = 7; // número, pois não está entre aspas simples 
System.out.printin(sete); // Não imprime nada!!!! 


Quando usando programas em outras línguas, às vezes queremos usar 
caracteres unicode, mas não temos um teclado com tais teclas (árabe, chinês 
etc.). Neste caso, podemos usar uma representação literal de um caractere 
unicode em nosso código, iniciando o char com Nu: 


char c = '\u03A9'; // unicode 
System.out.println(c); // imprime a letra grega ômega 


Identificadores 


Quando escrevemos nossos programas, usamos basicamente dois tipos 
de termos para compor nosso código: identificadores e palavras reservadas. 

Chamamos de identificadores as palavras definidas pelo programador 
para nomear variáveis, métodos, construtores, classes, interfaces etc. 

Já palavras reservadas ou palavras-chave são termos predefinidos da lin- 
guagem que podemos usar para definir comandos ( if, for, class, entre 
outras). 


São diversas palavras-chave na linguagem java: 
e abstract 
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assert 
boolean 
break 
byte 
case 
catch 
char 
class 
const 
continue 
default 
do 
double 
else 
enum 
extends 
false 
final 
finally 
float 
for 


goto 
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if 
implements 
import 
instanceof 
int 
interface 
long 
native 
new 

null 
package 
private 
protected 
public 
return 
short 
static 
strictfp 
super 
switch 
synchronized 


this 
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e throw 

e throws 
e transient 
e true 

e try 

e void 

e volatile 


e while 





NULL, FALSE E TRUE 


Outras três palavras reservadas que não aparecem nessa lista são 
true, falsee null. Mas, segundo a especificação na linguagem Java, 
esses três termos são considerados literais e não palavras-chave (embora 
também sejam reservadas), totalizando 53 palavras reservadas. 

http://java.sun.com/docs/books/tutorial/java/nutsandbolts/ 
-keywords.html 











Identificadores válidos devem seguir as seguintes regras: 


e Não podem ser igual a uma palavra-chave; 
e Podem usar letras (unicode), números, Se _; 
e O primeiro caractere não pode ser um número; 


e Podem possuir qualquer número de caracteres. 


Os identificadores são case sensitive, ou seja, respeitam maiúsculas e 
minúsculas: 
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int 
int 
int 
int 
int 
int 
int 
int 
int 
int 


umNome; // ok 

umnome ; // ok, diferente do anterior 

“num; // ok 

$ abc; // ok 

xy; // ok 

false; // inválido, palavra reservada 
x-y; // inválido, traço 

4num; // inválido, começa com número 
av#f; // inválido, # 


num.spc; // inválido, ponto no meio 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


1 


class A { 


public static void main(String[] args) { 
int 
idade 
= 100; 
System.out.println(idade); 


a) O código não compila: erros a partir da linha que define uma variável 


do tipo int. 


b) O código não compila: a variável idade não foi inicializada, mas foi 


usadaem System.out.println 


c) O código compila e imprime o. 


d) O código compila e imprime 100. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


i 


2 


class AÍ 
public static void main(Stringl] args) 1 
int idade; 
if (args.length > 0) 1 
idade = Integer .parseInt(args[0]); 
} else { 
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7 System.err.printin("Por favor passe sua idade como 
8 primeiro parâmetro"); 

9 } 

10 System.out.println("Sua idade é " + idade); 

nu + 

2 } 


a) Não compila: erro na linha que tenta acessar a variável idade. 
b) Compila e imprime o ou a idade que for passada na linha de comando. 
c) Compila e imprime a idade que for passada na linha de comando. 


d) Compila e imprime a mensagem de erro ou “Sua idade é "e a idade. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 
3 boolean array = new boolean[300]; 

4 System.out.printin(array [3]); 

5 } 

6} 


a) Imprime true. 
b) Imprime false. 
c) Imprime 0. 
d) Imprime -1. 
e) Imprime null. 


f) Não compila. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl[] args) 1 
3 boolean[] array = new boolean[300]; 
4 System.out.printin(array[3]); 

5 dg 

6} 
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a) Imprime true. 
b) Imprime false. 
c) Imprime 0. 
d) Imprime -1. 
e) Imprime null. 


f) Não compila. 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 boolean argumentos; 

4 if (args.length > 0) 

5 argumentos = 1; 

6 else 

7 argumentos = 0; 

8 System.out.println(argumentos) ; 

9 } 

10 + 


a) Não compila: o método de impressão não recebe boolean. 
b) Não compila: atribuição inválida. 
c) Não compila: o método length de array não é uma propriedade. 


d) Não compila: o método length de string[]não é uma pro- 
priedade. 


e) Compila e imprime 0 ou 1. 


f) Compila e imprime false ou true. 


6) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 int n = 09; 

4 int m = 03; 
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7) 


n 


8) 


9) 
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int x = 1 000; 
System.out.printin(x - n + m); 


a) Não compila: erro na linha que declara n. 
b) Não compila: erro na linha que declara x. 
c) Não compila: erro na linha que declara m. 


d) Compila e imprime um número menor que 1000. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(String[] args) { 
for(char c='a';c <= 'z';c++) { 
System.out.printin(c); 


a) Não compila: não podemos somar um em um caractere. 
b) Não compila: não podemos comparar caracteres com <. 


c) Compila e imprime o alfabeto entre a e z, inclusive. 


Qual das palavras a seguir não é reservada em Java? 


a) strictfp 
b) native 
c) volatile 
d) transient 


e) instanceOf 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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ı class A { 

2 public static void main(String[] args) { 
3 boolean BOOLEAN = false; 

4 if (BOOLEAN) {4 

5 System.out.printin("Sim"); 

6 } 

7 } 

s } 


a) Não compila: não podemos declarar uma variável com o nome de uma 
palavra reservada. 


b) Não compila: não podemos declarar uma variável iniciando com letras 
maiúsculas. 


c) Compila e roda, imprimindo Sim. 


d) Compila e roda, não imprimindo nada. 


4.2 DIFERENCIA ENTRE VARIÁVEIS DE REFERÊNCIAS A 
OBJETOS E TIPOS PRIMITIVOS 


As variáveis de tipos primitivos de fato armazenam os valores (e não pon- 
teiros/referências). Ao se atribuir o valor de uma variável primitiva a uma 
outra variável, o valor é copiado, e o original não é alterado: 


int a = 10; 

int b = a; // copiando o valor de a para b 
b++; // somando 1 em b 
System.out.printin(a); // continua com 10. 


Os programas construídos com o modelo orientado a objetos utilizam, 
evidentemente, objetos. Para acessar um atributo ou invocar um método de 
qualquer objeto, é necessário que tenhamos armazenada uma referência para 
o mesmo. 

Uma variável de referência é um ponteiro para o endereço de memória 
onde o objeto se encontra. Ao atribuirmos uma variável de referência a outra, 
estamos copiando a referência, ou seja, fazendo com que as duas variáveis 
apontem para o mesmo objeto, e não criando um novo objeto: 
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class Objeto 1 
int valor; 


class Teste{ 
public static void main(Stringl[] args) 
Objeto a = new Objeto(); 
Objeto b = a; // agora b aponta para o mesmo objeto de a 


a.valor = 5; 


System.out.printin(b.valor); // imprime 5 


Duas referências são consideradas iguais somente se elas estão apontando 
para o mesmo objeto. Mesmo que os objetos que elas apontem sejam iguais, 
ainda são referências para objetos diferentes: 


Objeto a = new Objeto(); 
a.valor = 5; 


Objeto b = new Objeto(); 
b.valor = 5; 


Objeto c = a; 


System.out.printin(a == b); // false 
System.out.printin(a == c); // true 


Veremos bastante sobre comparação de tipos primitivos e de referências 
mais à frente. 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 
3 int x = 15; 

4 int y = X; 


64 


Casa do Código Capítulo 4. Trablhando com tipos de dados em Java 





int z = y; 
2==5 
System.out.println(x + y + z); 


Imprime 43. 
Imprime 44. 
Imprime 45. 
Imprime 46. 
Imprime 47. 
Imprime 48. 


Imprime 49. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { 


2 


3} 


int v = 15; 


4 class A { 


5 


6 


public static void main(String[] args) { 


B x = new B(); 

By =x; 

y.v+t+; 

X.v++; 

Bz =y; 

Z.V--; 

System.out.println(x.v + y.v + z.v); 


a) Imprime 43. 


b) Imprime 44. 


c) Imprime 45. 
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d) Imprime 46. 
e) Imprime 47. 
f) Imprime 48. 


g) Imprime 49. 


4.3 LEIA OU ESCREVA PARA CAMPOS DE OBJETOS 


Ler e escrever propriedades em objetos é uma das tarefas mais comuns em 
um programa java. Para acessar um atributo, usamos o operador . (ponto), 
junto a uma variável de referência para um objeto. Veja a seguinte classe: 


class Carro { 
String modelo; 
int ano; 


public Carro() { ano = 2014; } 


public String getDadosDeImpressao() { 
return modelo + " - " + ano; 


public void setModelo(String m) 1 
this.modelo = m; 


Vamos escrever um código para usar essa classe: 


ı Carro a = new Carro(); 
2 a.modelo = "Palio"; // acessando diretamente o atributo 
3 a.setModelo("Palio"); // acessando o atributo por um método 


s // acessando o método e passando o retorno como argumento para 


6 // o método println 
7 System.out.printin(a.getDadosDeImpressao()); 
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As linhas 2 e 3 têm exatamente o mesmo efeito. Como iniciamos o valor 
da propriedade ano no construtor, ao chamar o método imprimeDados, o 
valor 2014 é exibido junto ao nome do modelo. 


Quando estamos dentro da classe, não precisamos de nenhum operador 
para acessar os atributos de instância da classe. Opcionalmente, podemos 
usar a palavra-chave this, que serve como uma variável de referência para 
o próprio objeto onde o código está sendo executado: 


class Carroí 
int ano; 
int modelo; 


public Carro (0 
modelo = "Indefinido"; // acessando variável de 
// instancia sem o this 
this.ano = 2014; // acessando com o this. 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class BÍ 

2 int c; 

3 void c(int c) { 

4 C= Ci 

5 j 

6 + 

7 class A { 

8 public static void main(Stringl] args) 1 
9 B b = new B(); 

10 b.c = 10; 

u System.out.println(b.c); 
12 b.c(30); 

3 System.out.println(b.c); 
14 } 

5 } 


a) Não compila: conflito de nome de variável membro e método em B. 
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b) Não compila: conflito de nome de variável membro e variável local em 
B. 


c) Compila e roda, imprimindo 10 e 30. 


d) Compila e roda, imprimindo outro resultado. 


4.4 EXPLIQUE O CICLO DE VIDA DE UM OBJETO (CRI- 
AÇÃO, “DEREFERÊNCIA” E GARBAGE COLLECTION) 


O ciclo de vida dos objetos java está dividido em três fases distintas. Vamos 
conhecê-las e entender o que cada uma significa. 


Criação de objetos 


Toda vez que usamos o operador new, estamos criando uma nova instân- 
cia de um objeto na memória: 


class Pessoa { 
String nome; 


class Teste 1 
public static void main(Stringl] args) 1 
Pessoa p = new Pessoa(); // criando um novo objeto do 
// tipo Pessoa 


Repare que há uma grande diferença entre criar um objeto e declarar uma 
variável. A variável é apenas uma referência, um ponteiro, não contém um 
objeto de verdade. 


// Apenas declarando a variável, 
// nenhum objeto foi criado aqui 
Pessoa p; 


// Agora um objeto foi criado e atribuído a variável 
p = new Pessoa(); 
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Objeto acessível 


A partir do momento em que um objeto foi criado e atribuído a uma var- 
iável, dizemos que o objeto está acessível, ou seja, podemos usá-lo em nosso 
programa: 


Pessoa p = new Pessoa(); // criação 
p.nome = "Mário"; // acessando e usando o objeto 


Objeto inacessível 


Um objeto é acessível enquanto for possível “alcançá-lo” através de al- 
guma referência direta ou indireta. Caso não exista nenhum caminho direto 
ou indireto para acessar esse objeto, ele se torna inacessível.: 


Pessoa p = new Pessoa(); 
p.nome = "Mário"; 


// atribuímos a p o valor null 
// o objeto não está mais acessível 
p = null 


// criando um objeto sem variável 
new Pessoa(); 


Nesse código, criamos um objeto do tipo Pessoa e o atribuímos à var- 
iável p. Na linha 6 atribuímos null a p. O que acontece com o objeto 
anterior? Ele simplesmente não pode mais ser acessado por nosso programa, 
pois não temos nenhum ponteiro para ele. O mesmo pode ser dito do objeto 
criado na linha 9. Após essa linha, não conseguimos mais acessar esse objeto. 

Outra maneira de ter um objeto inacessível é quando o escopo da variável 
que aponta para ele termina: 


int valor = 100; 
if( valor > 50) { 
Pessoa p = new Pessoa(); 
p.nome = "João"; 
+ // Após esta linha, o objeto do tipo Pessoa não está mais 
// acessível 
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Garbage Collector 


Todo objeto inacessível é considerado elegível para o garbage collector. Al- 
gumas questões da prova perguntam quantos objetos são elegíveis ao garbage 
collector ao final de algum trecho de código: 


public class Bla { 


int b; 
public static void main(Stringl[] args) { 
Bla b; 


for (int i = 0; i < 10; i++) { 
b = new Bla(); 
b.b = 10; 


System. out.println("fim"); 


Ao chegar na linha 9, temos 9 objetos elegíveis para o Garbage Collector. 





OBJETOS ELEGÍVEIS X OBJETOS COLETADOS 


O garbage collector roda em segundo plano juntamente com sua apli- 
cação java. Não é possível prever quando ele será executado, portanto 
não se pode dizer com certeza quantos objetos foram efetivamente co- 
letados em um certo ponto da aplicação. O que podemos determinar 
é quantos objetos são elegíveis para a coleta. A prova pode tentar se 
aproveitar do descuido do desenvolvedor aqui: nunca temos certeza de 
quantos objetos passaram pelo garbage collector, logo, somente indique 
quantos estão passíveis de serem coletados. 











Por fim, é importante ver um exemplo de referência indireta, no qual nen- 
hum objeto pode ser “garbage coletado”: 


ı import java.util.x*; 
2 class Carro { 


3 
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} 
class Carros { 
List<Carro> carros = new ArrayList<Carro>(); 
} 
class Teste { 
public static void main(String args[]) { 
Carros carros = new Carros(); 
for(int i = 0; i < 100; i++) 
carros.carros.add(new Carro()); 
// até essa linha todos ainda podem ser alcançados 


Nesse código, por mais que tenhamos criados 100 carros e um objeto do 
tipo Carros, nenhum deles pode ser garbage coletado pois todos podem ser 
alcançados direta ou indiretamente através de nossa thread principal. 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class BÍ 

3 5 

4 class A { 

5 public static void main(Stringl] args) 1 
6 B b; 

7 for(int i = 0;i < 10;i++) 

8 b = new B(); 

9 System.out.println("Finalizando!"); 
10 } 

11 } 


a) Não compila. 


b) Compila e garbage coleta 10 objetos do tipo B na linha do 


System. out. 


c) Compila e não podemos falar quantos objetos do tipo B foram garbage 
coletados na linha do System. out. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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ı class BÍ 

3 5 

4 class A { 

5 public static void main(Stringl] args) 1 
6 B b = new B(); 

7 for(int i = 0;i < 10;i++) 

8 b = new B(); 

9 System.out.println("Finalizando!"); 
10 } 

nu } 


a) Não compila. 
b) Compila e 10 objetos do tipo B podem ser garbage coletados ao chegar 
na linha do System.out. 


c) Compila e 11 objetos do tipo B podem ser garbage coletados ao chegar 
na linha do System. out. 


d) Compila e garbage coleta 1 objetos do tipo B na linha do 


System. out. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B{ 

35 

4 class A 1 

5 public static void main(Stringl] args) 1 
6 BL] bs = new B[100]; 

7 System.out.println("Finalizando!"); 
8 } 

ə } 


a) Compila e 100 objetos do tipo B são criados, mas não podemos falar 
nada sobre o garbage collector ter jogado os objetos fora na linha do 


System. out. 


b) Compila e nenhum objeto do tipo B é criado. 
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c) Compila, cria 100 e joga fora todos os objetos do tipo B ao chegar no 


System.out. 


4.5 CHAME MÉTODOS EM OBJETOS 


Além de acessar atributos, também podemos invocar métodos em um objeto. 
Para isso usamos o operador . (ponto), junto a uma variável de referência 
para um objeto. Deve-se prestar atenção ao número e tipo de parâmetros do 
método, além do seu retorno. Métodos declarados como void não possuem 
retorno, logo, não podem ser atribuídos a nenhuma variável ou passado para 


outro método como parâmetro: 


class Pessoal 
String nome; 
public String getNome 01 


return nome; 


public void setNome (String nome)f 
this.nome = nome; 


class Testel 
public static void main(String[] args) 
Pessoa p = new Pessoa(); 


//chamando método na variável de ref. 
p. setNome ("Mario"); 


//Atribuindo o retorno do método a variável. 
String nome = p.getNome(); 


// erro, método é void 
String a = p.setNome("X"); 
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Quando um método está sendo invocado em um objeto, podemos chamar 
outro método no mesmo objeto através da invocação direta ao nome do 
método: 


class A { 
void metodoi() { 


metodo2(); // chama o metodo2 no objeto onde metodol foi 


// chamado 
} 
void metodo2() { 
} 


Argumentos variáveis: varargs 


A partir do Java 5, varargs possibilitam um método que receba um 
número variável (não fixo) de parâmetros. É a maneira de receber um array 
de objetos e possibilitar uma chamada mais fácil do método. 

Um caso especial é quando método recebe um argumento variável ( 
varargs). Neste caso, podemos chamá-lo com qualquer número de argu- 
mentos: 


class Calculadora{ 
public int soma(int... nums){ 
int total = 0; 
for (int a : nums){ 
total+= a; 
} 


return total; 


nums realmente é um array aqui, você pode fazer um for usando o 
length, ou mesmo usaro enhanced for. A invocação desse método pode 
ser feita de várias maneiras: 
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public static void main (Stringl] args){ 
Calculadora c = new Calculadora(); 


//Todas as chamadas abaixo sao válidas 
System.out.printin(c.soma()); 
System.out.printin(c.soma(1)); 
System.out.printin(c.soma(1,2)); 
System.out.printin(c.soma(1,2,3,4,5,6,7,8,9)); 


Em todos os casos, um array será criado, nunca null será passado. 
Um parâmetro varargs deve ser sempre o último da assinatura do método 
para evitar ambiguidade. Isso implica que apenas um dos parâmetros de um 
método seja varargs. E repare que os argumentos variáveis têm que ser do 
mesmo tipo. 

E será dada a prioridade para o método que já podia existir antes no Java 
1.4: 


void metodo(int ... x) (5) 
void metodo(int x) {} 


metodo (5) ; 


Isso vai invocar o segundo método. Podemos também passar um array 
de ints para um método que recebe um varargs: 


void metodo(int ... x) {} 
metodo (new int [] (1,2,3,4)); 


Mas nunca podemos chamar um método que recebe array como se ele 


fosse varargs: 


void metodo(int[] x) { + 


metodo(1,2,3); // não compila 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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ı class B { 

2 void x() { 

3 System. out .println("vazio"); 

4 } 

5 void x(String... args) { 

6 System.out.printin(args.length); 
7 } 

3} 

ə class C { 

10 void x(String... args) { 

u System.out.printin(args.length); 
12 } 

5 void x() 1 

14 System. out .println("vazio"); 

15 E; 

16 } 

ız class AÍ 

18 public static void main(Stringl] args) 1 
19 new BO) .x(); 

20 new CO .x(); 

21 } 

2 + 


a) Não compila: conflito entre método com varargs esem argumentos. 
b) Compila e imprime vazio/vazio. 

c) Compila e imprime vazio/o. 

d) Compila e imprime o/vazio. 


e) Compila e imprime o/o. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B{ 

2 void x(int... x) É 

3 System.out.printin(x. length); 
4 } 

5} 

e class A { 
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7 public static void main(String[] args) { 
8 new B() .x(23789,673482); 

9 E; 

10 + 


a) Não compila: varargs tem método e não atributo length. 
b) Compila e ao rodar imprime os dois números. 


c) Compila e ao rodar imprime 2. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class BÍ 

2 void x(int... x) { 

3 System.out.println(x.length); 

4 } 

5 } 

6 class A { 

7 public static void main(String[] args) { 
8 new B() .x(new int []123789,673482)); 
9 } 

10 + 


a) Não compila: varargs tem método e não atributo length. 

b) Não compila: não podemos passar um array para um varargs. 
c) Compila e ao rodar imprime os dois números. 

d) Compila e ao rodar imprime 1. 


e) Compila e ao rodar imprime 2. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B{ 

2 void x(Object... x) { 

3 System.out.println(x.length); 
4 j 

5} 

é class A { 
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7 public static void main(String[] args) { 

8 new B() .x(new Object []123789,673482)); 
9 } 

10 + 


a) Não compila: varargs tem método e não atributo length. 

b) Não compila: não podemos passar um array para um varargs. 
c) Compila e ao rodar imprime os dois números. 

d) Compila e ao rodar imprime. 


e) Compila e ao rodar imprime 2. 


4.6 MANIPULE DADOS USANDO A CLASSE STRING- 
BUILDER E SEUS MÉTODOS 


Para suportar Strings mutáveis, o Java possui as classes StringBuffer e 
StringBuilder. A operação mais básica é o append que permite concate- 
nar ao mesmo objeto: 


StringBuffer sb = new StringBuffer(); 
sb.append("Caelum") ; 

sb.append(" - "); 

sb.append("Ensino e Inovação"); 


System.out.println(sb); // Caelum - Ensino e Inovação 


Repara que o append não devolve novos objetos como em String, mas 
altera o próprio StringBuffer, que é mutável. 
Podemos criar um objeto desse tipo de diversas maneiras diferentes: 


// vazio 

StringBuilder sb1 = new StringBuilder (); 

// conteudo inicial 

StringBuilder sb2 = new StringBuilder("java"); 

// tamanho inicial do array para colocar a string 
StringBuilder sb3 = new StringBuilder (50); 

// baseado em outro objeto do mesmo tipo 
StringBuilder sb4 = new StringBuilder (sb2); 


78 


Casa do Código Capítulo 4. Trablhando com tipos de dados em Java 





Tenha cuidado: ao definir o tamanho do array, não estamos criando uma 
String de tamanho definido, somente um array desse tamanho que será uti- 
lizado pelo StringBuilder, portanto: 


StringBuilder sb3 = new StringBuilder (50); 
System.out.printin(sb3); // linha em branco 
System.out.printin(sb3.length0); // 0 


Asclasses StringBuffere StringBuilder têm exatamente a mesma 
interface (mesmos métodos), sendo que a primeira é thread-safe e a última 
não (e foi adicionada no Java 5). Quando não há compartilhamento entre 
threads, use sempre que possívela StringBuilder, que é mais rápida por 
não precisar se preocupar com locks. 

Inclusive, em Java, quando fazemos concatenação de Strings usando o +, 
por baixo dos panos, é usado um StringBuilder. Não existe a operação 
+ na classe String. O compilador troca todas as chamadas de concatenação 
por StringBuilders (podemos ver isso no bytecode compilado). 


Principais métodos de StringBuffer e StringBuilder 


Há a família de métodos append com overloads para receber cada um 
dos primitivos, Strings, arrays de chars, outros StringBuffer etc. Todos 
eles devolvem o próprio StringBuffer/ Builder o que permite chamadas 
encadeadas: 


StringBuffer sb = new StringBuffer(); 
sb.append ("Caelum") .append(' - '') .append("Ensino e Inovação"); 
System.out.println(sb); // Caelum - Ensino e Inovação 


O método append possui uma versão que recebe Object e chama o 
método toString de seu objeto. 

Há ainda os métodos insert para inserir coisas no meio. Há versões 
que recebem primitivos, Strings, arrays de char etc. Mas todos têm o primeiro 
argumento recebendo o índice onde queremos inserir: 


StringBuffer sb = new StringBuffer(); 
sb.append("Caelum - Inovação"); 
sb.insert(9, "Ensino e "); 
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System.out.println(sb); // Caelum - Ensino e Inovação 


Outro método que modifica é o delete, que recebe os índices inicial e 
final: 


StringBuffer sb = new StringBuffer(); 
sb.append("Caelum - Ensino e Inovação"); 
sb.delete(6, 15); 


System.out.printin(sb); // Caelum e Inovação 


Para converter um StringBuffer/ Builder em String, basta 


chamar o toString mesmo. O método reverse inverte seu conteúdo: 


System.out.println(new StringBuffer ("guilherme") .reverse()); 
// emrehliug 


Fora esses, também há o trim, charAt, length(), equals, 
index0Of, last Index0Of, substring. 

Cuidado, pois o método substring não altera o valor do seu 
StringBuilder ou StringBuffer, mas retorna a String que você de- 
seja. Existe também o método subSequence que recebe o início e o fim e 
funciona da mesma maneira que o substring com dois argumentos. 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 

3 StringBuilder sb = new StringBuilder (); 
4 sb.append ("guilherme") .delete(2,3); 

5 System. out .println(sb); 

6 } 

7} 


a) O código não compila: erro na linha que tenta imprimir o 


StringBuilder 


b) O código compila e imprime glherme. 
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c) O código compila e imprime guherme. 


d) O código compila e imprime gilherme. 





e) O código compila e imprime gulherme. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 

3 StringBuilder sb = new StringBuilder ("guilherme"); 

4 System. out .println(sb.index0f("e") + sb.lastIndex0f("e")); 
5 System.out.printin(sb. index0f ("k'') + sb.lastIndex0f ("k')); 
6 } 
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a) O código imprime 13 e -2. 
b) O código imprime 13 e o. 
c) O código imprime 13 e -1. 
d) O código imprime 13 e 8. 
e) O código imprime 13 e 10. 
f) O código imprime 15 e -2. 
g) O código imprime 15 e o. 
h) O código imprime 15 e -1. 
i) O código imprime 15 e 8. 


j) O código imprime 15 e 10. 


4.7 CRIANDO E MANIPULANDO STRINGS 


Existem duas maneiras tradicionais de criar uma String: 


String nomeDireto = "Java"; 
String nomeIndireto = new String("Java"); 


A comparação entre esses dois tipos de criação de Strings é feita na 
seção Test equality between strings and other objects using == and equals() 5.3 


Existem outras maneiras não tão comuns: 
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char [] nome = new char[]{'J', 'a!, 'v', ta'}ł}; 
String nomeComArray = new String (nome); 


StringBuilder sbi = new StringBuilder("Java"); 
String nome1 = new String(sbl); 


StringBuffer sb2 = new StringBuffer ("Java"); 
String nome2 = new String(sb2); 


Como uma String não é um tipo primitivo, ela pode ter valor null, 
lembre-se disso: 


String nome = null; // null explicito 
Podemos concatenar Strings como +: 
String nome = "Certificação" + " " + "Java"; 


Caso tente concatenar null com uma String, temos a conversão de 
null para String: 


String nula = null; 
System.out.printin("nula: " + nula); // imprime nula: null 


O Java faz a conversão de tipos primitivos para Strings automatica- 
mente, mas lembre-se da precedência de operadores: 


String nome = "Certificação" + ' ! + "Java" + ! ' + 1500; 
System.out.println (nome); 


String nome2 = "Certificação"; 
nome2 += ! ! + "Java! + ! ' + 1500; 


System.out.println (nome2); 
String valor = 15 + 00 + " certificação"; 


System.out.printlin(valor); // imprime "15 certificação", 
// primeiro efetuando uma soma 
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Strings são imutáveis 
O principal ponto sobre Strings é que elas são imutáveis: 


String s = "caelum"; 
s.toUpperCase(); 
System.out.printin(s); 


Esse código imprime caelum em minúscula. Isso porque o método 
toUpperCase não altera a String original. Na verdade, se olharmos o 
javadoc da classe String vamos perceber que todos os métodos que pare- 
cem modificar uma String na verdade devolvem uma nova. 


String s = "caelum"; 
String s2 = s.toUpperCase(); 
System.out.println(s2); 





Agora sim imprimirá CAELUM, uma nova String. Ou, usando a mesma 
referência: 
String s = "caelum"; 


s = s.toUpperCase(); 
System.out.printin(s); 


Para tratarmos de “strings mutáveis”, usamos as classes StringBuffer 
e StringBuilder. 

Lembre-se que a String possui um array por trás e, seguindo o padrão 
do Java, suas posições começam em o: 


// 0=g, devolve 'g' 
char caracter0 = "guilherme". charAt(0); 


// 0=g 1=u, devolve 'u' 
char caracteri = "guilherme".charAt(1); 


// 0=g 1=u 2=i, devolve 'i' 
char caracter2 = "guilherme". charAt (2); 


Cuidado ao acessar uma posição indevida, você pode levar 





um — StringIndexOutOfBoundsException (atenção ao nome da 











Exception, não é Array IndexOutofBoundsException): 
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char caracter20 = "guilherme". charAt(20); // exception 
char caracterMenosUm = "guilherme". charAt(-1); // exception 


Principais métodos de String 
O método length imprime o tamanho da String: 


String s = "Java"; 

System.out.println(s.length()); // 4 

System.out.println(s.length); // não compila: não é atributo 

System.out.println(s.size()); // não compila: não existe size 
// em String Java 





Já o método isEmpty dizsea String tem tamanho zero: 


System.out.println("".isEmpty0); // true 
System. out.println("java".isEmpty()); // false 
System.out.println(" ".isEmpty0); // false 


Devolvem uma nova String: 
e String toUppercase () tudo em maiúscula; 
e String toLowercCase () tudo em minúsculo; 


e String trim() retira espaços em branco no começo e no fim; 


o 
[96] 


tring substring (int beginIndex, int endIndex) de- 


volve a substring a partir dos índices de começo e fim; 


e String substring (int beginIndex) semelhante ao anterior, 
mas toma a substring a partir do índice passado até o final da String; 


e String concat (String) concatena o parâmetro ao fim da String 
atual e devolve o resultado; 


e String replace (char oldChar, char newChar) substitui 
todas as ocorrências de determinado char por outro; 








e String replace (CharSequence target, CharSequence 


replacement) substitui todas as ocorrências de determinada 





CharSequence (como String) por outra. 
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O método trimlimpa caracteres em branco nas duas pontas da String: 


System.out.println(" " trim()); // imprime só a quebra de 
// linha do println 
System.out.println(" ".trim().isEmpty0); // true 
System.out.printin(" guilherme "); // imprime 'guilherme' 
System.out.printin(". ."); // imprime '. .' 


O método replace substituirá todas as ocorrências de um texto por 
outro: 


System.out.println('"java".replace("j', "J")); // Java 
System.out.println("guilherme".replace("e", "i")); // guilhirmi 


Podemos sempre fazer o chaining e criar uma sequência de “transfor- 
mações” que retornam uma nova String: 


String parseado = " Quero tirar um certificado oficial de 
Java! ''.toUpperCase() .trim(); 


// imprime: "QUERO TIRAR UM CERTIFICADO OFICIAL DE JAVA!" 
System.out.printin(parseado) ; 


Para extrair pedaços de uma String, usamos o método substring. 
Cuidado ao usar o método substring com valores inválidos, pois eles 





jogam uma Exception. O segredo do método susbtring é que ele não 
inclui o caractere da posição final, mas inclui o caractere da posição inicial: 


String texto = "Java"; 


// ava 
System.out.printin(texto.substring(1)); 


// StringIndex0ut0fBoundsException 
System.out.printin(texto.substring(-1)); 


// StringIndex0ut0fBoundsException 
System.out.printin(texto.substring(5)); 


// Java 
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System.out.printin(texto.substring(0, 4)); 


// ava 
System.out.printin(texto.substring(i, 4)); 


// Jav 
System.out.printin(texto.substring(0O, 3)); 


// StringIndex0ut0fBoundsException 
System.out.printin(texto.substring(0O, 5)); 


// StringIndex0ut0fBoundsException 
System.out.printin(texto.substring(-1, 4)); 


Comparação: 


* boolean equals (Object) compara igualdade caractere a carac- 
tere (herdado de Object); 





e boolean equalsIgnoreCase (String) compara caractere a car- 
actere ignorando maiúsculas/minúsculas; 


e int compareTo (String) compara as 2 Strings por ordem lexi- 
cográfica (vem de Comparable); 


e int compareToIgnoreCase (String) compara as 2 Strings por 
ordem lexicográfica ignorando maiúsculas/minúsculas. 


E aqui, todas as variações desses métodos. Não precisa saber o número ex- 
ato que o compareTo retorna, basta saber que será negativo caso a String 
na qual o método for invocado vier antes, zero se for igual, positivo se vier 
depois do parâmetro passado: 


String texto = "Certificado"; 
System.out.println(texto.equals("Certificado'")); // true 
System.out.println(texto.equals('"certificado")); // false 
System.out.println(texto.equalsIgnoreCase('certificado'));//true 


System.out.printin(texto.compareTo("Arnaldo")); // 2 
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System 
System 


System 


System 


.out.printin(texto. 
.out.printin(texto. 


.out.printin(texto. 


.out.printin(texto. 


Buscas simples: 


e bool 


compareTo("Certificado")); // 0 
compareTo("Grécia")); // -4 


compareTo("certificado")); // -32 


compareToIgnoreCase(''certificado"));//0 


lean contains (CharSequence) devolve true se a String 


contém a sequência de chars; 


e bool 





String do parâmetro; 


e bool 





String do parâmetro; 


lean startsWith (String) devolve true se começa com a 


lean endsWith (String) devolve true se termina com a 


e int indexof (char) e int indexoOf (String) devolve o índice 


da primeira ocorrência do parâmetro; 


e int lastIndexoOf (char) e int lastIndexoOf (String) de- 


volve o índice da última ocorrência do parâmetro. 


O código a seguir exemplifica todos os casos desses métodos: 


String texto = "Pretendo fazer a prova de certificação de Java"; 


System. 
System. 
System. 


System. 
System. 
System. 


System. 
System. 


System. 
System. 


out 
out 
out 


out 
out 
out 


out 
out 


out 
out 


.printin(texto. 
.printin(texto. 
.printin(texto. 


.printin(texto. 
.printin(texto. 
.printin(texto. 


.printin(texto. 
.printin(texto. 


.printin(texto. 
.printin(texto. 


index0f ("Pretendo')); // imprime O 
index0f ("Pretendia!)); // imprime -1 
index0f ("tendo")); // imprime 3 


index0f("a")); // imprime 10 
lastIndex0f("a'")); // imprime 45 


lastIndex0f ("Pretendia!)); //imprime -1 


startsWith("Pretendo")); // true 
startsWith("Pretendia'!)); // false 


endsWith("Java')); // true 
endsWith("Oracle'")); // false 
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1) Considere o seguinte código dentro de um main: 


class AM 
public static void main(String [] args){ 
String s = "aba"; 
for(int i = 0; i < 9; i++) { 
s = s +"aba"; 
} 
System.out.printin(s.length); 


a) Não compila. 

b) Compila e imprime 3. 
c) Compila e imprime 30. 
d) Compila e imprime 33. 


e) Compila e imprime 36. 


2) Dada a seguinte classe: 


class B { 
String msg; 


void imprime() { 
if (!msg.isEmpty()) 
System. out .println (msg); 
else 
System. out .println("vazio"); 


O que acontece se chamarmos new B(). imprime () ? 


a) Não compila. 
b) Compila, mas dá exceção na hora de rodar. 


c) Compila, roda e não imprime nada. 
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d) Compila, roda e imprime “vazio”. 


3) Dada a seguinte classe: 


class B { 


void imprime() { 
String msg; 
if (!msg.isEmpty()) 
System.out.println(msg) ; 
else 
System. out .println("vazio"); 


O que acontece se chamarmos new B ().imprime () ? 


a) Não compila. 
b) Compila, mas dá exceção na hora de rodar. 
c) Compila, roda e não imprime nada. 


d) Compila, roda e imprime “vazio”. 


4) Qual é a saída nos dois casos? 


String s = "Caelum"; 
s.concat(" - Ensino e Inovação"); 
System. out .println(s); 


StringBuffer s = new StringBuffer ("Caelum"); 
s.append(" - Ensino e Inovação"); 
System. out .println(s); 


a) ‘Caelum e ‘Caelum - Ensino e Inovação: 
b) ‘Caelum - Ensino e Inovação‘ e ‘Caelum - Ensino e Inovação. 
c) ‘Caelun e “Caelumí. 


d) ‘Caelum - Ensino e Inovação e “Caelumí 
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5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 String vazio = null; 

4 String full = "Bem-vindo " + vazio; 
5 System.out.printin(full); 

6 } 

7} 


a) Não compila pois vazio é nulo. 

b) Não compila por outro motivo. 

c) Compila e imprime “Bem-vindo “. 

d) Compila e imprime “Bem-vindo vazio”. 


e) Compila e imprime outro resultado que não foi mencionado nessas 
alternativas. 


6) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 String vazio; 

4 String full = "Bem-vindo " + vazio; 
5 System.out.printin(full); 

6 } 

7} 


a) Não compila pois vazio é nulo. 

b) Não compila por outro motivo. 

c) Compila e imprime “Bem-vindo “. 

d) Compila e imprime “Bem-vindo vazio”. 


e) Compila e imprime outro resultado que não foi mencionado nessas 
alternativas. 


7) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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ı class A { 

2 String vazio; 

3 public static void main(String[] args) { 
4 String full = "Bem-vindo " + vazio; 
5 System.out.printin(full); 

6 } 

7 + 


a) Não compila pois vazio é nulo. 

b) Não compila por outro motivo. 

c) Compila e imprime “Bem-vindo “ 

d) Compila e imprime “Bem-vindo vazio”. 

e) Compila e imprime outro resultado que não foi mencionado nessas 
alternativas. 


8) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 static String vazio; 

3 public static void main(Stringl] args) 1 
4 String full = "Bem-vindo " + vazio; 
5 System.out.printin(full); 

6 } 

z} 


a) Não compila pois vazio é nulo. 

b) Não compila por outro motivo. 

c) Compila e imprime “Bem-vindo “. 

d) Compila e imprime “Bem-vindo vazio”. 

e) Compila e imprime outro resultado que não foi mencionado nessas 
alternativas. 


9) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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ı class A { 

2 public static void main(Stringl] args) 1 
3 String s = null 

4 String s2 = new String(s); 

5 System.out.println(s2); 

6 } 

7} 


a) Não compila ao tentar invocar o construtor. 
b) Compila e não imprime nada. 
c) Compila e imprime null. 


d) Compila e dá erro de execução ao tentar criar a segunda String. 


10) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 

3 String s = "estudando para a certificação"; 
4 System.out.printin(s.substring(3, 6)); 

5 } 

6} 


a) Não compila: caractere com acento e cedilha dentro de uma String. 
b) Não compila: substringé subString. 

c) Compila e imprime “uda”, 

d) Compila e imprime “tuda”. 


e) Compila e imprime “tud”. 


11) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 String s2 = new String(null); 

4 System.out.println(s2); 

5 } 

6} 
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a) Não compila ao tentar invocar o construtor. 


b) Compila e não imprime nada. 


c) Compila e imprime null. 


d) Compila e dá erro de execução ao tentar criara String. 


12) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class AÍ 


2 


3 


4 


public static void main(Stringl] args) 1 


int valor = 10; 
int dividePor = 4; 
double resultado = valor / dividePor; 
System.out.println(valor + dividePor + 
" são os valores utilizados.'"); 


System.out.printin(resultado + " é o resultado"); 


a) Imprime os números 10, 4 € 2.5. 


b) Imprime os números 14 e 2.5. 


c) Nenhuma das outras alternativas. 


13) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class AÍ 


2 


public static void main(Stringl] args) 1 


String s = "estudando para a certificação"; 
s.replace('"e!!, 'a'); 
System.out.printin(s); 


a) Não compila. 


b) Compila e imprime “estudando para a certificação”. 


c) Compila e imprime “astudando para a cartificação”. 
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d) Compila e imprime “studando para a crtificação”, 


14) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 String s = "guilherme"; 

4 s.substring(0,2) = "gua"; 

5 System.out.printin(s); 

6 } 

7} 


a) Erro de compilação. 
b) Compila e imprime “guilherme”. 


b) 


c) Compila e imprime “gualherme”. 


94 


CAPÍTULO 5 


Usando operadores e construções 
de decisão 


5.1 USE OPERADORES JAVA 


Para manipular os valores armazenados das variáveis, tanto as primitivas 
quanto as não primitivas, a linguagem de programação deve oferecer oper- 
adores. Um dos operadores mais importantes é o que permite guardar um 
valor em uma variável. Esse operador é denominado operador de atribuição. 

No Java, o símbolo = representa o operador de atribuição. Para atribuir 
um valor precisamos de uma variável à qual será atribuído o valor, e do valor: 


long idade = ; // não compila, onde está o valor? 
long = 15; // não compila, onde está o nome da variável? 
long idade = 15; // compila 
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idade = 15; 
// compila desde que a variável tenha sido declarada 
// anteriormente 


Para um valor ser atribuído a uma variável, ambos devem ser compatíveis. 
Um valor é compatível com uma variável se ele for do mesmo tipo dela ou de 
um tipo menos abrangente. 


// Iniciando uma variável com o operador de atribuição "=". 
int idade = 10; 


// O valor literal 10 é do tipo int e a variável é do tipo long. 
// Como int é menos abrangente que long essa atribuição está 

// correta. 

long idade = 10; 


Procure sempre se lembrar dos tamanhos dos primitivos quando estiver 
fazendo a prova. int é um número médio, será que ele “cabe” em uma 
variável do tipo long (número grande)? Sim, logo o código compila. Mais 


exemplos: 
int a = 10; // tipos iguais 
long b = 20; // int cabe em um long 


float c = 10f; // tipos iguais 

double d = 20.0f; // float cabe em um double 

double e = 30.0; // tipos iguais 

float f = 40.0; // erro, double não cabe em um float. 
int g = 101; // erro, long não cabe em int 

float h = 101; // inteiros cabem em decimais 

double i = 20; // inteiros cabem em decimais 

long j = 20f; // decimais não cabem em inteiros 


A exceção a essa regra ocorre quando trabalhamos com tipos inteiros 
menos abrangentes que int (byte, short e char). Nesses casos, o com- 
pilador permite que atribuamos um valor inteiro, desde que compatível com 


o tipo: 
byte bi = 10; 
byte b2 = 200; // não compila, estoura byte 
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char ci 10; 


char c2 = -3; // não compila, char não pode ser negativo 
Atribuição e referência 


Quando trabalhamos com referências, temos que lembrar do polimor- 
fismo: 


List<String> lista = new ArrayList<String>0; 


E no caso do Java 7, quando atribuímos com generics podemos usar o 
operador diamante: 


// operador diamante na atribuição e inicialização: 
ArrayList<String> lista = new ArrayList<>(); 


// operador diamante e polimorfismo junto: 
List<String> lista = new ArrayList<>(); 


Lembre que as atribuições em Java são por cópia de valor, sempre. No 
tipo primitivo, copiamos o valor, em referências a objetos, copiamos o valor 
da referência (não duplicamos o objeto): 


List<String> lista = new ArrayList<>(); 
// copia o valor da referência, o objeto é o mesmo 
List<String> lista2 = lista; 


lista2.add("Guilherme''); 


// verdadeiro 
System.out.println(lista.size() == lista2.size()); 


int idade = 15; 


int idade2 = idade; // copia o valor 
idade2 = 20; 


System.out.printin(idade == idade2); // falso 
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Operadores aritméticos 


Os cálculos sobre os valores das variáveis primitivas numéricas são feitos 
com os operadores aritméticos. A linguagem Java define operadores para as 
principais operações aritméticas (soma, subtração, multiplicação e divisão). 


int dois = 2; 
int dez = 10; 


// Fazendo uma soma com o operador "+". 
int doze = dois + dez; 


// Fazendo uma subtração com o operador "-". 
int oito = dez - dois; 


// Fazendo uma multiplicação com o operador "*". 
int vinte = dois * dez; 


// Fazendo uma divisão com o operador "/". 
int cinco = dez / dois; 


Além desses, há um operador para a operação aritmética “resto da di- 
visão” Esse operador só faz sentido para variáveis primitivas numéricas in- 
teiras. 


int dois = 2; 
int dez = 10; 


// Calculando o resto da divisão de 10 por 2. 
int um = dez Y dois; 


O resultado de uma operação aritmética é um valor. A dúvida que surge 
é qual será o tipo dele. Para descobrir o tipo do valor resultante de uma op- 
eração aritmética, devem-se considerar os tipos das variáveis envolvidas. 

A regra é a seguinte: o resultado é do tipo mais abrangente entre os das 
variáveis envolvidas ou, no mínimo, o int. 


int idade 15: 


5; 


long anos 
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// ok, o maior tipo era long 
long daquiCincoAnos = idade + anos; 


// não compila, o maior tipo era long, devolve long 
int daquiCincoAnos2 = idade + anos; 


Mas devemos lembrar da exceção: o mínimo é um int: 


byte b = 1; 
short s = 2; 


// devolve no mínimo int, compila 
int i = b + sS; 


// não compila, ele devolve no mínimo int 
byte b2 = i + s; 


// compila forçando o casting, correndo risco de perder 


// informação 
byte b2 = (byte) (i + s); 


Divisão por zero 


Dividir (ou usar mod) um inteiro por zero lança uma 








ArithmeticException. Se o operando for um float ou double, 
isso gera infinito positivo ou negativo (depende do sinal do operador). As 
classes Float e Double possuem constantes para esses valores. 


int i 200; 


0; 


int v 


// compila, mas exception 
System.out.printin(i / v); 


// compila e roda, infinito positivo 
System.out.printin(i / 0.0); 


Ainda existe o valor NaN (Not a Number), gerado pela radiciação de um 
número negativo e por algumas contas com números infinitos. 
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100 / 0.0; 
-100 / 0.0; 


double positivoInfinito 


double negativoInfinito 


// número não definido (NaN) 
System.out.println(positivoInfinito + negativoInfinito); 


Comparadores 


A comparação entre os valores de duas variáveis é feita através dos oper- 
adores de comparação. O mais comum é comparar a igualdade e a desigual- 
dade dos valores. Existem operadores para essas duas formas de comparação. 


e == igual 


e != diferente 


Além disso, os valores numéricos ainda podem ser comparados em re- 
lação à ordem. 


e > maior 


e < menor 


° 
vV 
Il 


maior ou igual 


° 
A 
IH 


menor ou igual 


Uma comparação pode devolver dois valores possíveis: verdadeiro ou 
falso. No Java, uma comparação sempre devolve um valor boolean. 


System.out.printin(1 == 1); // true. 
System.out.printin(1 != 1); // false. 
System.out.printin(2 < 1); // false. 
System.out.printin(2 > 1); // true. 

System.out.printin(1 >= 1); // true. 
System.out.printin(2 <= 1); // false. 


Toda comparação envolvendo valores numéricos não considera o tipo do 
valor. Confira somente se eles têm o mesmo valor ou não, independente de 
seu tipo: 
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// true. 
System.out.printin(i == 1.0); 


// true. 
System.out.println(1 == 1); 


// true. 1.0 float é 1.0 double 
System.out.println(1.0f == 1.0d); 


// true. 1.0 float é 1 long 
System.out.println(1.0f == 11); 


Os valores não primitivos (referências) e os valores boolean devem ser 
comparados somente com dois comparadores, o de igualdade ( ==) e o de 
desigualdade ( !=). 


// não compila, tipo não primitivo só aceita != e == 
System.out.println("Mario" > "Guilherme"); 


// não compila, boolean só aceita != e == 
System.out.printin(true < false); 


Não podemos comparar tipos incomparáveis, como um boolean com 
um valor numérico. Mas podemos comparar chars com numéricos. 


// não compila, boolean é boolean 
System.out.printin(true == 1); 


// compila, 'a! tem valor numérico também 
System.out.println('a! > 1); 


Cuidado, é muito fácil comparar atribuição com comparação e uma pe- 
gadinha aqui pode passar despercebida, como no exemplo a seguir: 


int a = 5; 
System.out.printin(a = 5); // não imprime true, imprime 5 


101 


5.1. Use operadores Java Casa do Código 








PRECISÃO 


Ao fazer comparações entre números de ponto flutuante, devemos 
tomar cuidado com possíveis problemas de precisão. Qualquer conta 
com estes números pode causar um estouro de precisão, fazendo com 
que ele fique ligeiramente diferente do esperado. Por exemplo, 1 == 
(100.0 / 100) pode não ser verdadeiro caso a divisão tenha uma pre- 
cisão não exata. 











Operadores lógicos 


Muitas vezes precisamos combinar os valores booleans obtidos, por ex- 
emplo, com comparações ou diretamente de uma variável. Isso é feito uti- 
lizando os operadores lógicos. 

Em lógica, as operações mais importantes são: e, ou, ou exclusivo 


e negação. 


System.out.println(i == 1 & 1 > 2); // false. 


System. out.println(1 == 1 | 2 > 1); // true. 
System. out.println(1 == 1 ^ 2 > 1); // false. 
System.out.printin(!(1 == 1)); // false. 


Antes de terminar a avaliação de uma expressão, eventualmente, o re- 
sultado já pode ser descoberto. Por exemplo, quando aplicamos a operação 
lógica e, ao achar o primeiro termo falso não precisamos avaliar o restante 
da expressão. 

Quando usamos esses operadores, sempre os dois lados da expressão são 
avaliados mesmo nesses casos em que não precisariam. 

Para melhorar isso, existem os operadores de curto circuito && e ||. 
Quando já for possível determinar a resposta final olhando apenas para a 
primeira parte da expressão, a segunda não é avaliada: 


System. out.println(1 != 1 && 1 > 2); 
// false, o segundo termo não é avaliado. 
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System.out.printin(i == 1 || 2 > 1); 
// true, o segundo termo não é avaliado. 


A maior dificuldade com operadores de curto circuito é se a segunda parte 
causa efeitos colaterais (um incremento, uma chamada de método). Avaliar 
ou não (independente da resposta) pode influenciar no resultado final do pro- 
grama. 


public static boolean metodo(String msg) { 
System. out .println (msg); 
return true; 


public static void main(String[] args) { 
System.out.println(1 == 2 & metodo("oi")); 
// imprime oi, depois false 
System.out.println(1 == 2 && metodo("tchau")); 
// não imprime tchau, imprime false 


int i = 10; 

System.out.println(i == 2 & i++ == 0); 
// imprime false, soma mesmo assim 
System.out.println(i); 

// imprime 11 


int j = 10; 

System.out.println(j == 2 && j++ == 0); 
// imprime false, não soma 
System.out.println(j); 

// imprime 10 


Incrementos e decrementos 


Para facilitar a codificação, ainda podemos ter operadores que fazem cál- 
culos (aritméticos) e atribuição em uma única operação. Para somar ou sub- 
trair um valor em 1, podemos usar os operadores de incremento/decremento: 


int i = 5; 
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// 5 - pós-incremento, i agora vale 6 
System.out.printin(i++); 


// 6 - pós-decremento, i agora vale 5 
System.out.println(i--); 


/[/ 5 
System.out.printin(i); 


E incrementos e decrementos antecipados: 


int i = 5; 


System.out.printin(+i); // 6 - prê-incremento 
System.out.printin(--i); // 5 - prê-decremento 
System.out.println(i); // 5 


Cuidado com os incrementos e decrementos em relação a pré e pós. 
Quando usamos pós-incremento, essa é a última coisa a ser executada. E 
quando usamos o pré-incremento, é sempre a primeira. 


int i = 105 


// 10, primeiro imprime, depois incrementa 
System. out.println(i++); 


// 11, valor já incrementado. 
System. out.println(i); 


// 12, incrementa primeiro, depois imprime 
System.out.printin(+i); 


// 12, valor incrementado. 
System.out.printlin(i); 


Existem ainda operadores para realizar operações e atribuições de uma 
só vez: 


int a = 10; 
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// para somar 2 em a 
a=a+ 2; 


//podemos obter o mesmo resultado com: 
a += 2; 


//exemplos de operadores: 
int i = 5; 


i += 10; //soma e atribui 


System.out.println(i); // 15 
i -= 10; //subtrai e atribui 
System.out.println(i); // 5 


i *= 3; // multiplica e atribui 
System.out.println(i); // 15 


i /= 3; // divide a atribui 
System.out.println(i); 118 


i %= 2; // divide por 2, e atribui o resto 
System.out.println(i); da 


System.out.println(i+=3); // soma 3 e retorna o resultado: 4 


Nesses casos, o compilador ainda dá um desconto para operações com 
tipos teoricamente incompatíveis. Veja: 


byte bi = 3; // compila, dá um desconto 
bi = bi + 4; // não compila, conta com int devolve int 


byte b2 = 3; // compila, dá um desconto 
b2 += 4; // compila também, compilador gente boa! 


Esse último caso compila inclusive se passar valores absurdamente altos: 
b2+=400 é diferente de b2 = b2 + 400. Ele faz o casting e roda normal- 
mente. 
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Cuidado também com o caso de atribuição com o próprio autoincre- 
mento: 


int a = 10; 
a t= tta t a + +tras 


Como a execução é do primeiro para o último elemento das somas, temos 
as reduções: 


+= tra t a + tras 

= a t tta ta t tta; 

10 +11 + a + ++a; 

10 + 11 + 11 + tta; 

10 + 11 + 11 + 12; 

44; // a passa a valer 44 


y 2 


Um outro exemplo de operador pós-incremento, cujo resultado é1 e 2: 


int j = 0; 
(j++ * j + j++); 


int i 
System.out.printin(i); 
System.out.printlin(j); 


Pois: 


(O x j + j++); // j vale 1 
(0x 1 + j++); // j vale 1 
(O0 * 1 + 1); // j vale 2 
1; // j vale 2 


H H H H 


Podemos fazer diversas atribuições em sequência, que serão executadas 
da direita para a esquerda. O resultado de uma atribuição é sempre o valor 
da atribuição: 


int a = 15, b = 20, c = 30; 
a =b= c; // b = 30, portanto a = 30 


Outro exemplo mais complexo: 


int a = 15, b = 20, c = 30; 
a= (b=c +5) +5; // c= 30, portanto b = 35, portanto a = 40 
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Operador ternário Condicional 


Há também um operador para controle de fluxo do programa, como um 
if. É chamado de operador ternário. Se determinada condição acontecer, 
ele vai por um caminho, caso contrário vai por outro. 


A estrutura do operador ternário é a seguinte: 


variável = teste booleano ? valor se verdadeiro : valor se falso; 


int i = 5; 


System.out.println(i == 5 ? "verdadeiro": "falso");// verdadeiro 
System.out.printin(i != 5? 1: 2); // 2 
String mensagem = i % 2 == 0 ? "é par" : "é ímpar"; 


O operador condicional sempre tem que retornar valores que podemos 
usar para atribuir, imprimir etc. 


Operador de referência 


Para acessar os atributos ou métodos de um objeto precisamos aplicar o 
operador . (ponto) em uma referência. Você pode imaginar que esse oper- 
ador navega na referência até chegar no objeto. 


String s = new String("Caelum"); 


// Utilizando o operador "." para acessar um 
// objeto String e invocar um método. 
int length = s.length(); 


Concatenação de Strings 


Quando usamos Strings, podemos usar o + para denotar concatenação. 
É a única classe que aceita algum operador fora o ponto. 

Em Java, não há sobrecarga de operadores como em outras linguagens. 
Portanto, não podemos escrever nossas próprias classes com operadores di- 
versos. 
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STRINGBUILDER 


pilador resolve. No código compilado, na verdade, é usado um 
StringBuilder. 


A concatenação de Strings é um syntax sugar que o próprio com- 








Precedência 


Não é necessário decorar a precedência de todos operadores do Java, basta 


saber o básico, que primeiro são executados pré-incrementos/decrementos, 


depois multiplicação /divisão/mod, passando para soma/subtração, depois os 


shifts (<<, >>, >>>) e, por último, os pós-incrementos/decrementos. 


As questões da certificação não entram em mais detalhes que isto. 


Pontos importantes 
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Na atribuição de um valor para uma variável primitiva, o valor deve ser 
do mesmo tipo da variável ou de um menos abrangente. 


EXCEÇÃO À REGRA: Para os tipos byte, short e char, em 
atribuições com literais do tipo int, o compilador verifica se o valor a 
ser atribuído está no range do tipo da variável. 


Toda variável não primitiva está preparada somente para armazenar 
referências para objetos que sejam do mesmo tipo dela. 


Toda comparação e toda operação lógica devolve boolean. 


O resultado de toda operação aritmética é no mínimo int ou do tipo 
da variável mais abrangente que participou da operação. 


A comparação de valores numéricos não considera os tipos dos valores. 


As referências e os valores boolean só podem ser comparados com 
== 0 ls, 


Toda atribuição é por cópia de valor. 


Casa do Código Capítulo 5. Usando operadores e construções de decisão 





Observação: O recurso do autoboxing permite fazer algumas operações 
diferentes envolvendo variáveis não primitivas. Discutiremos sobre autobox- 
ing adiante. 


Casting de tipos primitivos 


Não podemos atribuir a uma variável de um tipo um valor que não é com- 
patível com ela: 


double d = 3.14; 
int i = d; 


Só podemos fazer essas atribuições se os valores forem compatíveis. Com- 
patível é quando um tipo cabe em outros, e ele só cabe se o range (alcance) 
dele for mais amplo que o do outro. 

byte -> short -> int -> long -> float -> double 

char -> int 

Se estivermos convertendo de um tipo que vai da esquerda para a direita 
nessa tabelinha, não precisamos de casting, a autopromoção fará o serviço 
por nós. 

Se estamos indo da direita para a esquerda, precisamos do casting e não 
importam os valores que estão dentro. Exemplo: 


double d 0; 
float f d; 


Esse código não compila sem um casting! O casting é a maneira que us- 
amos para moldar uma variável de um tipo em outro. Nós estamos avisando 
o compilador que sabemos da possibilidade de perda de precisão ou trunca- 
mento, mas nós realmente queremos fazer isso: 


float f = (float) d; 


Podemos fazer casting entre ponto flutuante e inteiro, o resultado será o 
número truncado, sem as casas decimais: 


double d = 3.1415; 
int i= (int) d; // 3 
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DICA 


Não é preciso decorar a sequência int->long->float etc. Basta lem- 
brar os alcances das variáveis. Por exemplo, o char tem dois bytes e 
guarda um número positivo. Será então que posso atribuir um char a 
um short? Não, pois um short tem 2 bytes, e usa meio a meio entre 
os números positivos e negativos. 











1) Qual código a seguir compila? 


a) short s = 10; 


char c = s; 
b) char c = 10; 
long 1 = c; 


c) char c = 10; 
short s = c; 


2) Faça contas com diferentes operandos: 


int ii = 3/2; 
double i2 = 3/2; 
double i3 = 3/2.0; 


long x = 0; double d = 0; 
double zero = x + d; 
System.out.println(il + i2 + i3 + x + d+ zero); 


Qual o resultado? 


a) 3 
b) 3.5 
c) 4 
d) 4.5 
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3) 


4) 





O código a seguir pode lançar um NullPointerException. Como 





evitar isso mantendo a mesma lógica? 


void metodo(Carro c) { 
if(c != null & c.getPreco() > 100000) { 
System.out.println("possivel sequestro"); 


a) Trocando != por == 
b) Trocando > por < 
c) Trocando & por | 


d) Trocando & por s& 


Alguns testes interessantes com tipos primitivos: 


int i = (byte) 5; 
long 1 = 3.0; 
float f = 0.0; 
char c = 3; 

char c2 = -2; 


Quais compilam? 


a) Lfec 
b) Lícecz 
c) i fec2 
d) iec 

e) fec 

f) fec2 

g) iel 

h) Lfec 


i) i cec2 
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5) A expressão a seguir pode ser reduzida, como podemos fazer? 


if ((trem && !carro) || (!trem && carro)) 1 
// 


e Trocando para usar um operador & eum | 

e Trocando para usar dois operadores « eum | 
e Trocando para usar um operador ! eum ^ 

e Trocando para usar um operador ^ 

* Removendo os parênteses 

e Removendo o | | do meio 


e Removendo os ! 


6) Imprima a divisão por o de números inteiros e de números com ponto 
flutuante: 


System.out.printin(3 / 0); 
System.out.printin(3 / 0.0); 
System.out.println(3.0 / 0); 
System.out.println(-3.0 / 0); 


Quais os resultados? 


7) class Xyz { 
2 public static void main(Stringl[] args) 1 


3 int y; 

4 for(int x = 0; x<10; ++x) { 
5 y=xh5+2; 

6 } 

7 System.out.println(y); 

8 + 

ə } 


Qual o resultado desse código? 
a) Erro de compilação na linha 3 
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b) Erro de compilação na linha 7 
c) ı 
d) 2 
e) 3 
f) 4 
g) 5 
h) 6 


8) class Teste { 


2 public static void main(String[] args){ 
3 byte b = 1; 
4 int i = 1; 
5 long 1 = 1; 
6 float f = 1.0; 
7 $ 
s } 
O código: 


a) Não compila a linha 3 pois “” é int e não pode ser colocado em um 
byte 

b) Não compila a linha 4 pois “” é long e não pode ser colocado em um 
int 


«>» 


c) Não compila a linha 5 pois “” é int e não pode ser colocado em um 





long 








d) Não compila a linha 6 pois “1.0” é double e não pode ser colocado 
em um float 


e) Todas as linhas compilam 


9) class $ 000 $ 1 
public static void main(Stringl] args) 1 


int $$ = 5; 
int . = $$++; 
if (__ < ++$$ II __-- > $$) 
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1) 
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System. out.print("A"); 


System. out.print ($$); 
System.out.print(. ); 


O estranho código: 


a) Não compila por causa do nome da classe 

b) Não compila por causa dos nomes das variáveis 
c) Compila mas dá erro na execução 

d) Compila, roda e imprime A76 

e) Compila, roda e imprime A75 

f) Compila, roda e imprime A74 


g) Compila, roda e imprime 76 


O que acontece com o seguinte código? Compila? Roda? 


public class Testel 
public static void main(Stringl] args) 1 


byte b1 = 5; 
byte b2 = 3; 
byte b3 = bi + b2; 


O que acontece com seguinte código? 


public class Testeí 
public static void main(Stringl] args) 1 
byte bi = 127; 
byte b2 -128; 
byte b3 = bi + b2; 
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6 System.out.printin(b3); 

7 } 

8 + 
a) Não compila por um erro na linha 3 
b) Não compila por um erro na linha 4 
c) Não compila por um erro na linha 5 


d) Compila e imprime -1 


1) public class Teste ( 


2 public static void main(String[] args) { 
3 int i; 

4 for (i = 0; i < 5; i++) { 

5 if (++i % 3 == 0) { 

6 break; 

7 } 

8 } 

9 System.out.printin(i); 

10 } 

11 } 


Qual é o resultado do código: 
a) imprime ı 
b) imprime 2 
c) imprime 3 
d) imprime 4 


2) Considerando o mesmo código da questão anterior, e se trocarmos para 


pós-incremento dentro do if? 


ı public class Teste { 


2 public static void main(String[] args) { 
3 int i; 
A for (i = 0; i < 5; i++) { 
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i if (i++ % 3 == 0) { 
6 break; 


8 } 
9 System.out.printlin(i); 


Qual é o resultado?: 


a) imprime 
b) imprime 2 
c) imprime 3 


d) imprime 4 


3) Qual é o resultado do seguinte código: 


public class Teste { 
public static void main(Stringl] args) 1 
int iz 
for (i = 0; i < 5; i+) { 
if (++i % 3 == 0) { 
break; 


F} 
System.out.printin(i); 


4) E se trocarmos o pré-incremento para pós-incremento ( i++)? 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 
3 byte bi = 100; 

4 byte b2 = 131; 
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5 System.out.printin(b1); 
6 } 
7 + 
a) Compila e imprime um número positivo. 
b) Compila e imprime um número negativo. 
c) Compila e dá uma exception de estouro de número. 
d) Compila e imprime um número que não sabemos dizer ao certo. 
e) Compila e imprime “Not A Number”. 


f) Não compila. 


6) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 char c = 65; 

4 char c2 = -3; 

5 System.out.printin(c + c2); 

6 } 

75 


a) Não compila nas duas declarações de char. 

b) Não compila nas três linhas dentro do método main. 
c) Não compila somente na declaração de c2. 

d) Não compila somente na soma de caracteres. 

e) Compila e roda, imprimindo 62. 


f) Compila e roda, imprimindo um outro valor. 


7) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 char c = 65; 

4 char c2 = 68 - 65; 
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8) 


n 


9) 


m 
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System.out.println(c + c2); 


a) Não compila nas duas declarações de char. 

b) Não compila nas três linhas dentro do método main. 
c) Não compila somente na declaração de c2. 

d) Não compila somente na soma de caracteres. 

e) Compila e roda, imprimindo 62. 


f) Compila e roda, imprimindo um outro valor. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(String[] args) { 
double resultado = 15 / 0; 
System.out.println(resultado); 


a) Não compila. 
b) Compila e dá exception. 
c) Compila e imprime positivo infinito. 


d) Compila e imprime o. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(Stringl] args) 1 
String resultado = "divisao dá: " + 15 / 0.0; 
System.out.println(resultado); 


a) Não compila. 
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b) Compila e dá exception. 
c) Compila e imprime positivo infinito. 


d) Compila e imprime o. 


10) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 System. out .println(1==true); 

4 + 

5 5 


a) Não compila. 
b) Compila e imprime verdadeiro. 


c) Compila e imprime falso. 


5.2 USE PARENTÊSES PARA SOBRESCREVER A PRE- 
CEDÊNCIA DE OPERADORES 


As vezes desejamos alterar a ordem de precedência de uma linha, e nesses 
instantes usamos os parênteses: 


int a 
int b 


15 *x4+ 1; // 15 * 4 = 60, depois 60 + 1 
15 x (4 + 1); //4+ 1 =5, depois 15 * 5 


61 
75 


Devemos tomar muito cuidado na concatenação de String e precedên- 
cia: 


System. out.println(15 + O + " é cento e cinquenta"); 
// 15 é cento e cinquenta 

System.out.println(15 + (0 + " é cento e cinquenta")); 
// 150 é cento e cinquenta 


System. out .println(("guilherme" + " silveira").length()); 
// 18 
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1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 

3 String resultado = ("divisao dá: " + 15) / 0.0; 
4 System.out.println(resultado); 

5 } 

6} 


a) Não compila. 
b) Compila e dá exception. 
c) Compila e imprime positivo infinito. 


d) Compila e imprime o. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 

3 System.out.printin(((! (true==false))==true ? 1 : 0)==0); 
4 } 

5 + 


a) Imprime true. 
b) Imprime false. 
c) Não compila. 

d) Imprime. 


e) Imprime o. 


5.3 TESTE A IGUALDADE ENTRE STRINGS E OUTROS 
OBJETOS USANDO == E EQUALS() 


Observe o seguinte código que cria duas Strings: 


ı String nome1 = new String("Mario"); 
2 String nome2 = new String("Mario'); 
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Como já estudamos anteriormente, o operador == é utilizado para com- 
paração. Neste caso, como se tratam de objetos, irá comparar as duas refer- 
ências e ver se apontam para o mesmo objeto: 


String nome1 = new String("Mario"); 


String nome2 = new String("Mario"); 


System.out.println(nome1 == nome2); // imprime false 


Até aqui tudo bem. Mas vamos alterar um pouco nosso código, mudando 
a maneira de criar nossas Strings, e rodar novamente: 


String nome1 = "Mario"; 
String nome2 = "Mario"; 
System. out .println(nome1 == nome2); // o que imprime? 


Ao executar o código, vemos que ele imprime true. O que aconteceu? 


Pool de Strings 


O Java mantém um pool de objetos do tipo String. Antes de criar uma 
nova String, primeiro o Java verifica neste pool se uma String com o mesmo 
conteúdo já existe; caso sim, ele a reutiliza, evitando criar dois objetos exata- 
mente iguais na memória. Como as duas referências estão apontando para o 
mesmo objeto do pool, o == retorna true. 

Mas por que isso não aconteceu antes, com nosso primeiro exemplo? O 
Java só coloca no pool as Strings criadas usando literais. Strings criadas com 
o operador new não são colocadas no pool automaticamente. 


String nome1 = "Mario"; //será colocada no pool 


String nome2 = new String("Mario"); 
/* 

"Mario" é colocado, mas nome2 é outra 
referência, não colocada no pool 


*/ 


Sabendo disso, temos que ter cuidado redobrado quando comparando 
Strings usando o operador ==: 
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String si = "string"; 

String s2 = "string"; 

String s3 = new String("string"); 

System.out.println(si == s2); // true, mesma referencia 
System.out.println(si == s3); // false, referências diferentes 


System.out.println(si.equals(s3)); // true, mesmo conteúdo 


Repare que, mesmo sendo instâncias diferentes, quando comparadas us- 
ando o método equals, o retorno é true, caso o conteúdo das Strings seja 
o mesmo. 

Quando concatenamos literais, a String resultante também será colocada 
no pool. 


String ab = "a" + "p"; 
System. out.println("ab" == ab); // true 


Mas isso é verdade apenas usando literais em ambos os lados da con- 
catenação. Se algum dos objetos não for um literal, o resultado será um novo 
objeto, que não estará no pool: 


String a = "a"; 
String ab = a + "b"; //usando uma referência e um literal 
System.out.printin("ab" == ab); // false 


Sabemos que Strings são imutáveis, e que cada método chamado em uma 
String retorna uma nova String, sem alterar o conteúdo do objeto original. 
Esses objetos resultantes de retornos de métodos não são buscados no pool, 
são novos objetos: 


String str = "um texto qualquer"; 
String txti 
String txt2 = x.substring(3, 8); //cria uma nova string 
System.out.println(txtl == txt2); // false 
System.out.println(txt.equals(x.substring(3, 8)); // true 


"texto"; 
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OS MÉTODOS DE STRING SEMPRE CRIAM NOVOS OBJETOS? 


Nem sempre. Se o retorno do método for exatamente o conteúdo at- 
ual do objeto, nenhum objeto novo é criado: 


String str = "HELLO WORLD"; 


4 String upper = str.toUpperCase(); // já está maiúscula 
String subs = str.substring(0,11); // string completa 

4 System.out.printin(str == upper); // true 
System.out.println(str == subs); // true 
System.out.printin(str == str.toString()); // true 











Contando Strings 


Uma questão recorrente na prova é contar quantos objetos do tipo 
String são criados em um certo trecho de código. Veja o código a seguir 
e tente descobrir quantos objetos String são criados: 


String h = new String ("hello "3; 
String hi = "hello "; 
String w = "world"; 


System.out.printin("hello "); 
System.out.printin(hi + "world"); 
System.out.println("Hello " == h1); 


E então? Vamos ver passo a passo: 


//Cria 2 objetos, um literal (que vai para o pool) e o outro 
//com o new 
String h = new String ("hello "); 


//nenhum objeto criado, usa o mesmo do pool 
String h1 = "hello "; 

//novo objeto criado e inserido no pool 
String w = "world"; 
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4 public static void main(String[] args) { 
for(int i = 0; i< 10; i++) 

| System.out .printin (metodo ()); 

| + 

4 

1 private static String metodo() { 
String x = "x"; 

d return x.toString(); 

1 + 

1} } 
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//nenhum objeto criado, usa do pool 
System.out.println("hello "); 


//criado um novo objeto resultante da concatenação, 
// mas este não vai para o pool 


System.out.printin(hi + "world"); 


//Novo objeto criado e colocado no pool (Hello com H maiúsculo). 
System.out.printin("Hello " == h1); // i 


Logo temos 5 Strings criadas. 





CUIDADO COM STRING JÁ COLOCADAS NO POOL 


Para descobrir se uma String foi criada e colocada no pool é 
necessário prestar muita atenção ao contexto do código e ao enunciado 
da questão. A String só é colocada no pool na primeira execução do tre- 
cho de código. Cuidado com questões que criam Strings dentro de méto- 
dos, ou que dizem em seu enunciado que o método já foi executado pelo 
menos uma vez: 


| public class Testes { 


Ao executar essa classe, apenas um objeto String será criado. O único 
lugar onde a String é criada é na linha 8 do código. 
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O método equals 


Para comparar duas referências, podemos sempre usar o operador ==. 
Dada a classe Cliente: 


class Cliente { 
private String nome; 
Cliente(String nome) 1 
this.nome = nome; 


} 
} 
Cliente c1 = new Cliente("guilherme"); 
Cliente c2 = new Cliente("mario"); 


System.out.println(ci==c2); // false 
System.out.println(ci==c1); // true 


Cliente c3 = new Cliente("guilherme'"'); 

System. out .println(ci==c3); 

// false, pois não é a mesma 

// referência: são objetos diferentes na memória 


Para comparar os objetos de uma outra maneira, que não através da refer- 
ência, podemos utilizar o método equals, cujo comportamento padrão é 
fazer a simples comparação com o ==: 


Cliente c1 = new Cliente("guilherme"); 
Cliente c2 
System. out .println(c1.equals(c2)); // false 
System.out.printin(ci.equals(ci)); // true 


new Cliente("mario"); 


Cliente c3 = new Cliente("guilherme''); 
System.out.printin(ci.equals(c3)); 

// false, pois não é a mesma 

// referência: são objetos diferentes na memória 


Isso é, existe um método em Object que você pode reescrever 
para definir um critério de comparação de igualdade. Classes como 
String, Integer e muitas outras possuem esse método reescrito, as- 
sim new Integer (10) == new Integer (10) dá false, mas new 





Integer (10) .equals (Integer (10)) dá true 
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É interessante reescrever esse método quando você julgar necessário um 
critério de igualdade diferente que o == retorna. Imagine o caso de nosso 


Cliente: 


class Cliente { 
private String nome; 
Cliente(String nome) 1 
this.nome = nome; 


public boolean equals(Object o) { 
if (! (o instanceof Cliente)) { 
return false; 
} 
Cliente outro = (Cliente) o; 
return this.nome.equals (outro.nome) ; 


O método equals não consegue tirar proveito do generics, então 
precisamos receber Object e ainda verificar se o tipo do objeto passado 
como argumento é realmente uma Cliente (o contrato do método diz que 
você deve retornar false, e não deixar lançar exception em um caso desses). 
Agora sim, podemos usar o método equals como esperamos: 


Cliente ci 
Cliente c2 = new Cliente("mario"); 


new Cliente('guilherme!'); 


System.out.println(ci.equals(c2)); // false 
System.out.println(ci.equals(ci)); // true 


Cliente c3 = new Cliente("guilherme'"); 
System.out.println(ci.equals(c3)); // true 


Cuidado ao sobrescrever o método equals: ele deve ser público, e 
deve receber Object. Caso você receba uma referência a um objeto do 
tipo Cliente, seu método não está sobrescrevendo aquele método padrão 
da classe Object, mas sim crianod um novo método (overload). Por 
polimorfismo o compilador fará funcionar neste caso pois o compilador fará a 
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conexão ao método mais específico, entre Object e Cliente, ele escolherá 
o método que recebe Cliente: 


class Cliente { 
private String nome; 
Cliente(String nome) 1 
this.nome = nome; 


public boolean equals(Cliente outro) { 
return this.nome.equals (outro .nome) ; 


} 


Cliente c1 = new Cliente("guilherme"); 
Cliente c2 = new Cliente("mario"); 
System.out.printin(ci.equals(c2)); // false 
System.out.printin(ci.equals(ci)); // true 


Cliente c3 = new Cliente("guilherme''); 
System.out.printin(ci.equals(c3)); // true 
System.out.printin(ci.equals((Object) c3)); 

// false, o compilador não sabe que Object é cliente, 
// invoca o equals tradicional, e azar do desenvolvedor 


Mas caso você use alguma biblioteca (como a API de coleções e de 
ArrayList do Java), o resultado não será o esperado. 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 
3 String s1 = "s1"; 

4 String s2 = "s" + "1"; 

5 System.out.println(sl==s2); 

6 System.out.println(sl==("" + s2)); 

7 } 

s } 


a) Não compila. 
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3) 


n 
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b) Compila e imprime true, false. 
c) Compila e imprime true, true. 


d) Compila e imprime false, false. 





e) Compila e imprime false, true. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class AÍ 
public static void main(Stringl] args) 1 
String s1 = "s1"; 
String s2 = si.substring(0, 1) + si.substring(t,1); 
System.out.println(sl==s2); 
System. out .println(s1.equals(s2)); 


a) Não compila. 
b) Compila e imprime true, false. 
c) Compila e imprime true, true. 


d) Compila e imprime false, false. 





e) Compila e imprime false, true. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(String[] args) { 
String s1 = "s1"; 
String s2 = s1.substring(0, 2); 
System.out.println(sl==s2); 
System.out.println(si.equals(s2)); 


a) Não compila. 


b) Compila e imprime true, false. 
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c) Compila e imprime true, true. 


d) Compila e imprime false, false. 





e) Compila e imprime false, true. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B extends C{} 


2 class C { 

3 int x; 

4 public boolean equals(C c) { 

5 return c.x==x; 

6 J 

75 

s class A { 

9 public static void main(Stringl] args) 1 
10 Ca = new C(); 

u C b= new B(); 

12 a.x = 1; 

13 b.x = 1; 

14 System.out.println(a==b); 

15 System.out.println(a.equals(b)); 
16 } 

7 } 


a) Não compila. 
b) Compila e imprime true, false. 
c) Compila e imprime true, true. 


d) Compila e imprime false, false. 





e) Compila e imprime false, true. 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B extends C{} 


2 class D { 
3 int x; 
45 
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s class CÍ 

6 int xX; 

7 public boolean equals (Object c) { 

8 return c.X==X; 

9 } 

10 + 

u class A { 

2 public static void main(Stringl] args) 1 
5 C a= new CC); 

14 C b= new D(); 

15 a.x = 1; 

16 b.x = 1; 

17 System.out.printin(a==b); 

18 System.out.printin(a.equals(b)); 
19 E; 

20 + 


a) Não compila. 
b) Compila e imprime true, false. 
c) Compila e imprime true, true. 


d) Compila e imprime false, false. 





e) Compila e imprime false, true. 


5.4 UTILIZE O IF E IF/ELSE 


Imagine um programa que aceita comandos do usuário, ou seja, um sistema 
interativo. De acordo com os dados que o usuário passar, o programa se com- 
porta de maneiras diferentes e, consequentemente, pode dar respostas difer- 
entes. 

O programador, ao escrever esse programa, deve ter recursos para 
definir o comportamento para cada possível comando do usuário, em out- 
ras palavras, para cada situação. Com isso, o programa será capaz de tomar 
decisões durante a execução com o intuito de mudar o fluxo de execução. 

As linguagens de programação devem oferecer aos programadores 
maneiras para controlar o fluxo de execução dos programas. Dessa forma, os 
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programas podem tomar decisões que afetam a sequência de comandos que 
serão executados. 


if / else 


A maneira mais simples de controlar o fluxo de execução é definir que um 
determinado trecho de código deve ser executado quando uma condição for 
verdadeira. 

Por exemplo, suponha um sistema de login. Ele deve verificar a autentici- 
dade do usuário para permitir ou não o acesso. Isso pode ser implementado 
com um if/else do Java. 


boolean autentico = true; 
if (autentico) { 
System. out .println("Usuario aceito"); 
} else { 
System. out .println("Usuario incorreto"); 


A sintaxe do if éa seguinte: 


if (CONDICAO) { 
// CODIGO 1 
} else { 
// CODIGO 2 


A condição de um if sempre tem que ser um valor booleano: 


ı if (1 - 2) { } // erro, numero inteiro 
3 if (1 < 2) {} //ok, resulta em true 


s boolean valor = true; 
6 if (valor == false) {} // ok, mas resulta em false 


s if (valor) {} // ok, valor é boolean 


Atenção dobrada ao código a seguir: 
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int a = 0, b= 1; 


if(a = b) { 
System.out.println("iguais"); 


Esta é uma pegadinha bem comum. Repare que não estamos fazendo uma 
comparação aqui, e sim, uma atribuição (um único =). O resultado de uma 
atribuição é sempre o valor atribuído, no caso, um inteiro. Logo, este código 
não compila, pois passamos um inteiro para a condição do if. 

A única situação em que um código assim poderia funcionar é caso a 
variável atribuída seja do tipo boolean, pois o resultado da atribuição será 


boolean: 


boolean a = true; 


if (a = false) { 
System.out.println("Falso!"); 


Neste caso, o código compila, mas não imprime nada. Após a atribuição, 
o valor da variável a é false,eo if não é executado. 
Caso só tenhamos um comando dentro do if ou else, as chaves são 


opcionais: 


if (!resultado) 
System.out.println("Falso!"); 

else 
System.out.println("Verdadeiro!''); 


Caso não tenhamos nada para ser executado em caso de condição false, 
não precisamos declarar o else: 


boolean autentico = true; 
if (autentico) 
System.out.println("Usuario aceito"); 


Mas sempre temos que ter algum código dentro do if, se não o código 
não compila: 
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boolean autentico = true; 
if (autentico) 
else // erro 
System.out.printin("Acesso negado"); 


Na linguagem Java, não existe o comando elseif. Para conseguir o efeito 
do “elseif”, os ifs são colocados dentro dos else. 


if (CONDICAO1) 1 
// CODIGO 1 

} else if (CONDICAO2) { 
// CODIGO 2 

} else { 
// CODIGO 3 


Grande parte das perguntas sobre estruturas de if/else são pegadin- 
has, usando a indentação como forma de distração: 


boolean autentico = true; 
if (autentico) 
System. out .println("Usuario aceito"); 
else 
System. out .println("Usuario incorreto"); 
System.out.println("Tente novamente"); 


A mensagem "Tente novamente" sempre é impressa, independente 
do valor da variável autentico. 

Esse foi um exemplo bem simples, vamos tentar algo mais complicado. 
Tente determinar o que é impresso: 


int valor = 100; 
if (valor > 200) 
if (valor <400) 
if (valor > 300) 
System. out.println("a"); 
else 
System. out.println("b"); 
else 
System.out.println("c"); 
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E então? "c"? Vamos reindentar o código para ver se fica mais fácil: 


int valor = 100; 
if (valor > 200) 
if (valor <400) 
if (valor > 300) 
System.out.printin("a''); 
else 
System.out.printin("b''); 
else 
System.out.printlin('"'c''); 


É sempre complicado analisar código não indentado ou mal indentado, 
e esse recurso é usado extensivamente em várias questões durante a prova, 
fique esperto! 


Unreachable Code e Missing return 


Um código Java não compila se o compilador perceber que aquele código 
não será executado sob hipótese alguma: 


class Teste 1 
public int metodo() 1 
return 5; 
System.out.println("Quando isso será executado?"); 


} 


Teste.java:10: unreachable statement 
System.out.println("Quando isso será executado?"); 


O código após o return não será nunca executado. Esse código não 
compila. Vamos ver alguns outros exemplos: 


class Teste { 
public int metodo(int x) { 
if(x > 200) 1 
return 5; 
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Este também não compila. O que será retornado se x for <= 200? 


Teste. java:12: missing return statement 


} 


1 error 


Vamos modificar o código para que ele compile: 


class Teste { 
public int metodo(int x) { 
if(x > 200) { 
return 5; 


} 


throw new RuntimeException(); 


Apesar de não estarmos retornando nada caso o if seja falso, o Java 
percebe que nesse caso uma exceção será disparada. A regra é: todos os cam- 
inhos possíveis devem retornar o tipo indicado pelo método, ou lançar ex- 
ceção. 


Em um if, essa expressão compila normalmente: 
if(false) {.... } //compila, apesar de ser unreachable code 
São pequenos detalhes, tome cuidado para não cair nessas pegadinhas. 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 

3 if (args.length > 0) 

4 System.out.printin("Um ou mais argumentos"); 
5 else 

6 System. out.println("0"); 

7 } 

s } 
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2) 


1 
2 
3 
4 


5; 


3) 


1 
2 
3 
4 


5 


a) Não compila: length é método. 
b) Não compila: faltou chaves no if e else. 
c) Se invocarmos sem argumentos, imprime 0. 


d) Nunca imprimirá 0. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B{ 
final boolean valor = false; 
} 
class A { 
public static void main(String[] args) { 
B b = new B(); 
if(b.valor = true) { 
System. out .println("verdadeiro"); 
} 
} 
} 


a) Não compila. 
b) Compila e imprime verdadeiro. 


c) Compila e não imprime nada.. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(Stringl] args) 1 
int quantidade = 15; 
if (quantidade=15) { 
System.out.printin("sim'"); 
} else { 
System. out.println("nao"); 


a) Não compila. 
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b) Imprime sim. 


c) Imprime não. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class AÍ 


2 


3 


4 


5 


public static void main(Stringl] args) 1 


if (args. length==1) 
System.out.println("Um'"); 
elseif (args. length==2) 
System.out.printin("Dois"); 
elseif (args. length==3) 
System.out.printin("Três"); 
else 
System.out.printin("Quatro"); 


a) Não compila. 


b) Roda e imprime “Um” quando passamos um argumento. 


c) Roda e imprime “Três” quando passamos 4 argumentos. 


d) Roda e não imprime nada quando passamos nenhum argumento. 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class AÍ 


2 


3 


4 


5 


public static void main(Stringl] args) 1 


String nome = args[0]; 

if (nome. equals("guilherme")) 
System.out.println (nome) ; 
System.out.printin("bom'"); 

else 
System.out.println('melhor ainda"); 
System.out.println (nome) ; 
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a) Erro de compilação no if. 
b) Erro de compilação no else. 


c) Compila e imprime o nome e “bom” caso o primeiro argumento seja 


guilherme. 


d) Compila e dá erro de execução caso não passe nenhum argumento na 
linha de comando. 


5.5 UTILIZE O SWITCH 


Suponha que um programa tenha que reagir diferentemente para três casos 
possíveis. Por exemplo, suponha que o usuário possa passar três valores pos- 
síveis: 1, 2e 3. Sefor 1, o programa deve imprimir “PRIMEIRA OPCAO’, 
se for 2, SEGUNDA OPCAO” e se for 3, “TERCEIRA OPCAO”, 

Isso pode ser implementado com if/else. Mas, há uma outra possi- 
bilidade. O Java, assim como outras linguagens de programação, oferece o 
comando switch. Ele permite testar vários casos de uma maneira diferente 
do if/else 


int opcao = 1; 
switch (opcao) { 


case 1: 

System.out.printin("PRIMEIRA OPCAO"); 
case 2: 

System.out.println("SEGUNDA OPCAO"); 
case 3: 


System.out.printin("TERCEIRA OPCAO"); 


O switch tem uma sintaxe cheia de detalhes e uma semântica pouco 
intuitiva. Vamos analisar cada um desses detalhes separadamente para ficar 
mais simples. 

O argumento do switch dever ser uma variável compatível com o tipo 
primitivo int, um wrapper de um tipo menor que Integer, uma String ou 
um enum. Enums Não são cobrados nessa prova, então vamos focar apenas 
nos outros dois casos. 


138 


Casa do Código Capítulo 5. Usando operadores e construções de decisão 





O valor de cada case deve ser compatível com o tipo do argumento do 
switch, caso contrário será gerado um erro de compilação na linha do case 
inválido. 


//argumento do switch int, e cases int 
int valor = 20; 
switch (valor){ 
case 10 : System.out.printin(10); 
case 20 : System.out.printin(20); 


//Argumento String, e cases String 

String s = "0i"; 

switch (s) 1 
case "Qi": System.out.printin("0lá"); 
case "Hi": System.out.printin("Hello"); 


//Argumento Byte, e cases byte 
Byte b = 10; 
switch (b) 1 
case 10: System.out.printin("DEZ''); 


//argumento do switch int, e cases string, não compila 
int mix = 20; 
switch (mix) 

case "10" : System.out.println(10); 

case "20" : System.out.println(20); 


Cuidado pois switch de double não faz sentido conforme a lista de 
argumentos que citamos compatíveis com o switch! 


double mix = 20; 

switch (mix){ // não compila 
case 10.0 : System.out.println(10); 
case 20.0 : System.out.println(20); 
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Você pode usar qualquer tipo primitivo menor que um int como argu- 
mento do switch, desde que os tipos dos cases sejam compatíveis: 


//argumento do switch byte 
byte valor = 20; 


switch (valor){ 
// Apesar de ser inteiro, 10 cabe em um byte, o compilador 
// fará o cast automaticamente 
case 10 
System.out.printIn(10); 


switch (valor){ 
// Neste caso, o número é muito grande, o compilador não 
// fará o cast e teremos um erro de compilação pois os tipos 
// são incompatíveis 
case 32768 : //erro 
System.out.printin(10); 


Em cada case, só podemos usar como valor um literal, uma variável 
final atribuída com valor literal, ou expressões envolvendo os dois. Nem 
mesmo null é permitido: 


int valor = 20; 
final int CINCO = 5; 
int trinta = 30; 


switch (valor) { 

case CINCO: // constante 
System.out.printin(5); 

case 10: // literal 
System.out.printin(10); 

case CINCO x 4: // operação com constante e literal 
System.out.println(20); 

case trinta: // erro, variável 
System. out.println(30); 

case trinta + CINCO: //erro, operação envolvendo variável 
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System.out.println(35); 
case null: // erro, null em case 
System. out .println("null"); 





CONSTANTES EM CASES 


Para ser considerada uma constante em um case, a variável, além 
de ser final, também deve ter sido inicializada durante a sua declaração. 
Inicializar a variável em outra linha faz com que ela não possa ser usada 
como valor em um case: 


int v = 10; 

final int DEZ = 10; 

final int VINTE; // final, mas não inicializada 
VINTE = 20; // inicializada 


switch (v) { 

case DEZ: 
System.out.println("DEZ!"); 
break; 

case VINTE: //erro 
System.out.println("DEZ!"); 
break; 











O switch também aceita a definição de um caso padrão, usando a 
palavra default. O caso padrão é aquele que deve ser executado se nen- 
hum case “bater”. 


int opcao = 4; 
switch (opcao) { 
case 1: 
System. out .println("PRIMEIRA OPCAO"); 
case 2: 
System. out.println("SEGUNDA OPCAO"); 
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case 3: 

System.out.printIn("TERCEIRA OPCAO"); 
default: 

System.out.printIn("CASO PADRAO"); 


Um detalhe sobre a sintaxe do default é que ele pode aparecer antes de 
um ou de diversos cases. Desta forma: 


int opcao = 4; 
switch(opcao) 1 
case 1: 
System.out.printin("PRIMEIRA OPCAO"); 
case 2: 
System.out.println("SEGUNDA OPCAO"); 
default: 
System.out.printIn("CASO PADRAO"); 
case 3: 
System.out.printIn("TERCEIRA OPCAO"); 


Um comportamento contraintuitivo do switch é que, quando execu- 
tado, se algum case “bater”, tudo que vem abaixo é executado também, todos 
os caseseo default, se ele estiver abaixo. Esse comportamento também 
vale se cair no default. Por exemplo, o código anterior imprime: 


CASO PADRAO 
TERCEIRA OPCAO 


Com esse comportamento, podemos inclusive criar cases sem nenhum 
bloco de código dentro: 


int v = 1; 
switch(v){ 
case 1: 
case 2: 
case 3: 
System.out.println("Até 3"); 
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Para mudar esse comportamento e não executar o que vem abaixo de um 
case que bater ou do default, é necessário usar o comando break em 


cada case. 


int opcao = 4; 
switch(opcao) 1 
case 1: 
System. out .println("PRIMEIRA OPCAO"); 
break; 
case 2: 
System. out .println("SEGUNDA OPCAO"); 
break; 
default: 
System.out.printIn("CASO PADRAO"); 
break; 
case 3: 
System. out .println("TERCEIRA OPCAO"); 
break; 


Neste caso, só será impresso “TERCEIRA OPCAO”. 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 int tamanho = args.length; 

4 switch(tamanho) { 

5 case 1: 

6 System. out.println("1"); 

7 case 2: 

8 System. out.println("2"); 

9 default: 

10 System.out.println('"mais argumentos"); 
n } 

12 } 

5 } 


a) Não compila. 
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b) Ao rodar sem argumentos joga uma exception.. 


«>» 


c) Ao rodar com dois argumentos, imprime somente “2”. 


d) Ao rodar com 5 argumentos, imprime “mais argumentos”. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 
3 int tamanhoEsperado = 1; 

4 int tamanho = args.length; 

5 switch (tamanho) 1 

6 case tamanhoEsperado: 

7 System. out.println("1"); 

8 break; 

9 default: 

10 System.out.printin(''cade o argumento?''); 
u } 

12 } 

5 5 


a) Não compila. 
b) Ao rodar sem argumentos joga uma exception. 


c) Ao rodar com um argumento, imprime somente “r”. 


d) Ao rodar com 5 argumentos, imprime “cade o argumento?” 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 

3 switch("Guilherme") { 

4 case "Guilherme": 

5 System.out.printin("Guilherme"); 
6 break; 

7 case "42"; 

8 System. out .println("42"); 

9 default: 
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4) 


1 


2 


5) 


System.out.printin("Qutro nome"); 


a) Não compila, pois um número não pode ser comparado com String. 
b) Compila e imprime Guilherme. 


c) Não compila, pois o código do case 42 e default nunca serão 


executados. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(Stringl] args) 1 
int count = args.length; 
switch(count) { 
case 0 { 
System. out .println("nenhum") ; 
break; 
} case 1 { 
} case 2 { 
System. out.println("ok"); 
} default { 
System. out .println("default"); 


a) Erro de compilação. 
b) Se rodar com 1 argumento, imprime ok e mais uma mensagem. 
c) Se rodar com 1 argumento, não imprime nada. 


d) Se rodar com 5 argumentos, imprime default. 


e) 
Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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ı class A { 

2 public static void main(Stringl] args) 1 
3 switch(10) { 

4 case < 10: 

5 System. out .println("menor"); 
6 default: 

7 System. out .println("igual"); 
8 case > 10: 

9 System.out.printin("maior'); 
10 E; 

nu } 

2 } 


a) Erro de compilação. 
b) Compila e imprime “igual”. 


c) Compila e imprime “igual” e “maior”. 


6) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 switch(10) { 

4 case 10: 

5 System. out.println("a"); 
6 break; 

7 System. out.println("b"); 
8 default: 

9 System.out.printin(''c''); 
10 case ll: 

u System. out.println("d"); 
12 } 

13 } 

u > 


a) Não compila. 
b) Imprime ae be ce a. 


c) Imprime a. 
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Criando e usando arrays 


6.1 DECLARE, INSTANCIE, INICIALIZE E USE UM ARRAY 
UNI-DIMENSIONAL 


As linguagens de programação, normalmente, fornecem algum recurso para 
o armazenamento de variáveis em memória sequencial. No Java, os arrays 
permitem esse tipo de armazenamento. 

Um array é um objeto que armazena sequencialmente “uma porção” de 
variáveis de um determinado tipo. É importante reforçar que os arrays são 
objetos. Uma referência para um objeto array deve ser armazenada em uma 
variável do tipo array. 

A prova de certificação verifica se o candidato está apto a manipular tanto 
arrays de tipos primitivos quanto de tipos não primitivos. 


Os quatro pontos importantes sobre arrays são: 
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e Declarar 
e Inicializar 
e Acessar 


e Percorrer 


Arrays de tipos primitivos 


Declaração: 
Para declarar um array, é utilizado [] logo após ao tipo das variáveis que 


desejamos armazenar ou logo após ao nome da variável. 


// Declaração de um array para guardar variáveis do tipo int. 
int[] idades; 


// Declaração de um array para guardar variáveis do tipo double. 
double pesos[]; 


// Declaração de um array para guardar variáveis do tipo long. 
long []pesos; 


// Declaração de um array para guardar variáveis do tipo long. 
long [l tamanhos; 


// Perceba as formas de declarar um array. 


Inicialização: 

Como um array é um objeto, a inicialização envolve a criação de um ob- 
jeto. O new, operador que cria objetos, é utilizado para construir um array. 
Se você não executa o new, qual o valor padrão? Para atributos, é null, e para 
variáveis locais, não há valor, como qualquer outra variável de referência: 


public class Clientes 1 
int[] idades; 


public static void main(Stringl] args) 1 
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Clientes c = new Clientes(); 
System.out.printin(c.idades); // imprime null 


} 


public class Produtos { 


public static void main(String[] args) { 
int[] precos; 
System.out.println(precos); // nao compila, não foi 
// inicializada 


E como instancio um array? 


// Inicialização do array idades. 
idades = new int [10]; 


// Inicialização do array pesos. 
pesos = new double[50]; 


Na inicialização, é definida a capacidade do array, ou seja, a quantidade de 
variáveis que ele terá. Quando falarmos em tamanho de um array, estaremos 
nos referindo à sua capacidade. 

Cada variável guardada em um array é iniciada implicitamente no mo- 
mento em que o array é criado. Os valores atribuídos às variáveis são os val- 
ores default. 


// Imprime O pois esse é o valor default para int. 
System.out.printin(idades[0]); 


E temos alguns casos extremos: 


//compila e roda 
int [] numeros = new int [0]; 


//compila, mas joga NegativeArraySizeException 
numeros = new int[-1]; 
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Durante a declaração de uma referência para um array, temos a oportu- 
nidade de criá-lo de uma maneira mais fácil se já sabemos o que queremos 
colocar dentro: 


int numeros; 
numeros = new int[]{1, 2, 3, 4, 5}; 


Carro[] carros = new Carro[] (new Carro(), null, new Carro(); 


Não passamos o tamanho e fazemos a declaração dos elementos entre 
chaves e separados por vírgula. Os arrays terão tamanho 5 e 3, respectiva- 
mente. 

E se a declaração e a inicialização estiverem na mesma linha podemos 
simplificar ainda mais: 


int [] numeros = (1, 2,3, 4,5); 


Mas temos que tomar um pouco de cuidado com esse modo mais simples 
de declarar o array. Só podemos fazer como no exemplo anterior quando 
declaramos e inicializamos o array na mesma linha. Se fizermos a declaração 
e a inicialização em linhas separadas, o código não compila: 


int [] numeros = (1, 2, 3, 4, 5}; //compila 
int [] numeros2; 
numeros2 = (1, 2, 3, 4, 5}; // não compila 


Se desejamos inicializar posteriormente, devemos adicionar o operador 


new para poder iniciar o array em outra linha: 


int [] numeros2; 
numeros2 = new int [] (1, 2, 3, 4, 5}; // compila 


Acesso 


As posições de um array são indexadas (numeradas) de 0 até a capacidade 
do array menos um. Para acessar uma das variáveis do array, é necessário 
informar sua posição. 
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// Coloca o valor 10 na primeira variável do array idades. 
int idades [] = new int [10]; 
idades[0] = 10; 


// Coloca o valor 73.14 na última variável do array pesos. 
double pesos[] = new double[50]; 
pesos[49] = 73.14; 


O que acontece se alguém tentar acessar uma posição que não existe? 


// Erro de execução ao tentar acessar um posição que não existe. 
// Array Index0Qut0fBoundsException 
pesos[50] = 88.4; 


Será gerado um erro de execução (não de compilação). A exception 





lançada pelo Java é Array IndexOutOfBoundsException 


Percorrendo 


Supondo que a capacidade de um array qualquer seja 100, os índices 
desse array variam de 0 até 99, ou seja, de 0 até a capacidade menos um. 

O tamanho de um array é definido na inicialização e fica guardado no 
próprio array, podendo ser recuperado posteriormente. 

Para recuperar o tamanho ou a capacidade de um array, é utilizado um 
atributo chamado length presente em todos os arrays. 


for (int i = 0; i < idades.length; i++) 1 
idades[il = i; 


No for tradicional, as posições de um array são acessadas através dos 
índices. Dessa forma, é possível, inclusive, modificar os valores que estão 
armazenados no array. 

Porém, em determinadas situações, é necessário apenas ler os valores de 
um array sem precisar modificá-los. Nesse caso, pode ser utilizado o for 
introduzido na versão 5 do Java. 


for(int idade : idades){ 
System.out.printin(idade); 
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Não há índices no for do Java 5. Ele simplesmente percorre os valores. 
Assim ele não permite modificar o array facilmente. 


Array de referências 


Em cada posição de um array de tipos não primitivos é guardada uma 
variável não primitiva. Esse é um fato fundamental. 


// Declarando e iniciando um array de Prova 
Proval] provas = new Proval[10]; 


Lembrando que o new inicia as variáveis implicitamente e que o valor 
padrão para variáveis não primitivas é null, todas as dez posições do array 
desse código estão null imediatamente após o new. 


// Erro de execução ao tentar aplicar o operador "." 
// em uma referência com valor null. 

// NullPointerException 

provas[0] .tempo = 10; 


Para percorrer um array de tipos não primitivos, podemos utilizar um 


laço: 


for (int i = 0; i < provas.length; i++){ 
provas[i] = new Prova(); 
provas[i].tempo = 210; 


for (Prova prova : provas){ 
System.out.println(prova.tempo); 


Caso a classe Prova seja abstrata, devido ao polimorfismo é possível adi- 
cionar filhas de Prova nesse array: o polimorfismo funciona normalmente, 


portanto funciona igualmente para interfaces. 


class Prova { 


} 


class ProvaPratica extends Prova { 


} 
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class Test { 
public static void main(Stringl[] args) { 
Proval] provas = new Proval[2]; 
provas[0] = new Prova(); 
provas[1] = new ProvaPratica(); 


Uma vez que o array de objetos é sempre baseado em referências, lembre- 
se que um objeto não será copiado, mas somente sua referência passada: 


Cliente guilherme = new Cliente(); 
guilherme. setNome("guilherme'') ; 


Cliente[] clientes = new Clientes[10]; 
clientes[0] = guilherme; 


System.out.printin(guilherme.getNome()); // guilherme 
System.out.printin(clientes[0].getNome()); // guilherme 


guilherme. setNome ("Silveira"); 


System.out.println(guilherme.getNome()); // silveira 
System.out.printin(clientes[0].getNome()); // silveira 


Casting de arrays 


Não há casting de arrays de tipo primitivo, portanto não adianta tentar: 


int [] valores = new int[10]; 
long[] vals = valores; // não compila 


Já no caso de referências, por causa do polimorfismo é possível fazer a 
atribuição sem casting de um array para outro tipo de array: 


String[] valores = new Stringl[2]; 
valores [0] = "Certificação"; 
valores[1] = "Java"; 


Object [] vals = valores; 
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for(Object valor : vals) { 
System.out.println(valor); // Certificação e depois Java 


E o casting compila normalmente mas, ao executarmos, um array de 





Object não é um array de String e levamos uma ClassCastException: 


Object [] valores = new Object [2]; 
valores[0] 


"Certificação"; 

valores[1] = "Java"; 

String[] vals = (Stringl[]) valores; 

for(Object valor : vals) 1 
System.out.printin(valor); 


Isso pois a classe dos dois é distinta e a classe paide array de string 
não é um array de objeto, e sim, um Object (lembre-se: todo array herda de 
Object): 


Object[ ] objetos = new Object[ 2 ]; 

Stringl ] strings = new Stringl 2 ]; 
System.out.println(objetos.getClass() .getName()); 
// [ Ljava.lang. Object; 
System.out.println(strings.getClass() .getName()); 
// [ Ljava.lang.String; 


System.out.println(strings.getClass() .getSuperclass()); 
// java.lang.Object 


1) Escolha a opção que não compila: 











a) int x; 
b) int xl]; 
c) int []x; 

d) int [] x; 
e) int x: 
f) [int x; 
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2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class À É 

2 public static void main(String[] args) { 
3 int x[] = new int[30]; 

4 int y[] = new int[3] {0,3,5}; 

5 } 

6} 


a) A primeira linha não compila. 
b) A segunda linha não compila. 


c) O código compila e roda. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir, 


em relação as linhas dentro do método main: 


ı class A { 

2 public static void main(String[] args) { 
3 int x[] = new int[0]; 

4 int x[] = new int[] {0,3,5}; 

5 int x[] = {0,3,5}; 

6 } 

7 > 


a) A primeira e segunda linhas não compilam. 

b) A segunda e terceira linhas não compilam. 

c) Somente a terceira linha não compila. 

d) O programa compila e roda, dando uma exception. 


e) O programa compila e roda, imprimindo nada. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A É 

2 public static void main(Stringl] args) 1 

3 int x[] = new int[3]; 

4 for(int i=x.length;i>=0;i--) x[il=ix2; 
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5 System.out.println("Fim!'"'); 
6 } 
7 > 

a) O programa não compila 

b) O programa imprime Fim. 


c) O programa compila e dá erro em execução. 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl[] args) 1 
3 int x[] = new int[3]; 

F for(x[1]=x.length-1;x[0]==0;x[1]--) { 
5 x[x[1]]=-5; 

6 System.out.printin(x[1]); 

7 } 

8 } 

ə} 


a) Não compila. 





b) Compila, imprime alguns números e dá uma Exception. 
c) Compila e não imprime nada. 

d) Compila e imprime 2. 

e) Compila e imprime -5. 

f) Compila e imprime 2, -5. 

g) Compila e imprime 2, -5, -5. 

h) Compila e imprime 2, 1, -5. 

i) Compila e imprime -5, -5. 


j) Dá exception. 


6) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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ı class A { 


2 


public static void main(Stringl] args) 1 


int x[] = new int[3]; 

for(x[1]=x.length-1;x[1]>=0;x[1]--) { 
x[x[1]]=-5; 
System.out.printin(x[1]); 


Não compila. 





Compila, imprime alguns números e dá uma Exception. 
Compila e não imprime nada. 

Compila e imprime 2. 

Compila e imprime -5. 

Compila e imprime 2, -5. 

Compila e imprime 2, —5, -5. 

Compila e imprime 2, 1, -5. 

Compila e imprime -5, -5. 


Dá exception. 


7) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class AÍ 


2 


3 


4 


5 


public static void main(String[] args) { 


String[] valores = new String[2]; 

valores[0] = "Certificação"; 

valores[1] = "Java"; 

Object[] vals = (Object[]) valores; 

vals[1] = "Daniela"; 

System.out.printin(vals[1] .equals(valores[1])); 


a) O código não compila. 
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b) O código compila e dá erro em execução. 


c) O código compila e imprime false. 


d) O código compila e imprime true. 














8) 

um array? 

a) int array = new int 
b) int array[] = new int 
c) int array = new int 
d) int array[] = new int 
e) int array = new int 
f) int array = new int 
g) int array = int [10]; 
h) int array = new int 
i) int array [] = new int 
j) int array[] = (1, 2, 3 





Quais das maneiras adiante são declarações e inicializações válidas para 





6.2 DECLARE, INSTANCIE, INICIALIZE E USE UM AR- 


RAY MULTI-DIMENSIONAL 


Podemos generalizar a ideia de array para construir arrays de duas dimensões, 


em outras palavras, array de arrays. 
de quantas dimensões quisermos. 


Declaração: 


// Um array de duas dimensões. 
int[] [] tabela; 


// Um array de três dimensões. 
int[] 0] cubol]; 
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// Um array de quatro dimensões. 
int [] [] []hipercubol]; 


// Perceba que as dimensões podem ser definidas do lado 
// esquerdo ou direito da variável. 


Inicialização: 


// Inicializando a primeira dimensão com 10 e a segunda com 15 
tabela = new int [10] [15]; 


// Inicializando a primeira dimensão com 10 e deixando as outras 
// para serem iniciadas depois 
cubo = new int [10] 0 0]; 


// Inicializando com valores 
int [][] teste = new int[] [1111,2,3),13,2,1),11,1,1)); 


Acesso: 


// Acessando a posição (0,1) 
System.out.printin(tabela[0] [1]); 


Podemos criar um array que não precisa ser “quadrado”, ele pode ter 
tamanhos estranhos: 


int [][] estranha = new int [2] [1]; 
estranha[0] = new int [20]; 
estranha[1] new int [10]; 
for(int i=0;i<estranha.length;i++) { 
System. out .println(estranha[i].length); // imprime 20 e 10 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 int zyx[] []=new int[3]; 

4 int[]x=new int [20]; 
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5 int []y=new int [10]; 

6 int []z=new int [30]; 

7 zyx[0]=x; 

8 zyx[1]=y; 

9 zyx[2]=z; 

10 System.out.printin(zyx[2].length); 
nu } 

2 + 


a) Não compila, erro ao declarar zyx. 

b) Compila e dá erro ao tentar atribuir o segundo array a zyx. 
c) Compila e dá erro ao tentar imprimir o tamanho do array. 
d) Compila e imprime 10. 

e) Compila e imprime 20. 


f) Compila e imprime 30. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 
3 int zyx[] []=new int [3] []; 

4 int []x=new int [20]; 

5 int []y=new int [10]; 

6 int []z=new int [30]; 

7 zyx[0]=x; 

8 zyx[1]=y; 

9 zyx[2]=z; 

10 System.out.printin(zyx[2].length); 
11 } 

2 + 


a) Não compila, erro ao declarar zyx. 
b) Compila e dá erro ao tentar atribuir o segundo array a zyx. 
c) Compila e dá erro ao tentar imprimir o tamanho do array. 


d) Compila e imprime 10. 
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e) Compila e imprime 20. 


f) Compila e imprime 30. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 
3 int zyx[] []=new int [3] [10]; 

4 int[|x=new int [20]; 

5 int []y=new int [10]; 

6 int[]z=new int [30]; 

7 zyx[0]=x; 

8 zyx[1]=y; 

9 zyx[2]=z; 

10 System.out.printin(zyx[2].length); 
11 } 

n2 + 


a) Não compila, erro ao declarar zyx. 
b) Não compila, erro ao atribuir arrays de tamanho diferente de 10 em 
ZYX. 
c) Compila e dá erro ao tentar atribuir o segundo array a zyx. 
d) Compila e dá erro ao tentar imprimir o tamanho do array. 
e) Compila e imprime 10. 
f) Compila e imprime 20. 
g) Compila e imprime 30. 
4) class Teste 1 
public static void main(Stringl[] args) 
int[] idades = new int [10]; 
idades [0] = 1.0; 
int [10][10] tabela = new int [10] [10]; 


int [][][] cubo = new int ĝ OC; 
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a) O código não compila. 
b) O código compila e dá erro em execução. 


c) O código compila e roda. 


Compila? Roda? 


6.3 DECLARE E USE UMA ARRAYLIST 


Nesta prova, veremos somente a ArrayList, uma lista que usa internamente 
um array. Rápida no método get, pois sua estrutura interna permite acesso 
aleatório (random access) em tempo constante. 


Jamais se esqueça de importar a ArrayList: 
import java.util.ArrayList; 

O primeiro passo é criar uma ArrayList vazia de Strings: 
ArrayList<String> nomes = new ArrayList<String>(); 


A ArrayList herda diversos métodos abstratos e concretos e veremos 
vários deles aqui, dentre esses, os principais para a certificação, vindos da 
interface Collection. 


Por exemplo, para adicionar itens, fazemos: 


ArrayList<String> nomes = new ArrayList<String>(; 
nomes .add('"certificação!"!); 
nomes .add("java!'); 


Para remover e verificar a existência do mesmo na lista: 


ArrayList<String> nomes = new ArrayList<String>0; 
nomes .add('"certificação!!); 
nomes .add("java!); 


System.out.println(nomes.contains('java')); // true 
System.out.println(nomes.contains('ct')); // false 


// true, encontrado e removido 
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boolean removido = nomes.remove('java"); 


System.out.println(nomes.contains("java')); // false 
System.out.println(nomes.contains("ct')); // false 


Note que o remove remove somente a primeira ocorrência daquele ob- 
jeto. 


Podemos também verificar o tamanho de nossa ArrayList: 


ArrayList<String> nomes = new ArrayList<String>(); 
nomes.add('"certificação!!); 

nomes.add("java!'); 
System.out.println(nomes.size()); // imprime 2 


E convertê-la para um array: 


ArrayList<String> nomes = new ArrayList<String>(); 
nomes.add('"certificação!!); 
nomes.add("java!'); 


Object [] nomesComoString = nomes.toArray(); 


Caso desejarmos um array de String, devemos indicar isso ao método 
toArray de duas formas diferentes: 


ArrayList<String> nomes = new ArrayList<String>(); 
nomes.add("certificação!!); 
nomes.add("java!); 


String[] nomes2 = nomes.toArray (new String[0]); 


String[] nomes3 = nomes.toArray (new String [nomes .size()]); 


Ambas passam um array de String: o primeiro menor e o segundo com o 
tamanho suficiente para os elementos. Se ele possui o tamanho suficiente, ele 
mesmo será usado, enquanto que, se o tamanho não é suficiente,o toArray 
cria um novo array do mesmo tipo. 


Além disso, podemos adicionar uma coleção inteira em outra: 
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ArrayList<String> nomes = new ArrayList<String>0; 
nomes .add('"certificação!"!); 
nomes .add("java!'); 


ArrayList<String> paises = new ArrayList<String>(); 
paises.add('"'coreia"); 
paises.add("brasil'"); 


ArrayList<String> tudo = new ArrayList<String>(); 
tudo.addAll (nomes); 

tudo .addAll (paises); 
System.out.printin(tudo.size()); // imprime 4 


Outros métodos são específicos da interface List e recebem uma 
posição específica onde você quer colocar ou remover algo do array usado 
na ArrayList. O método get devolve o elemento na posição desejada, 
lembrando que começamos sempre com o: 


ArrayList<String> nomes = new ArrayList<String>(); 
nomes .add('"certificação!!); 
System.out.println(nomes.get(0)); // imprime certificação 


Já o método add foi sobrecarregado para receber a posição de inclusão: 


ArrayList<String> nomes = new ArrayList<String>(; 
nomes .add('"certificação!!); 
System.out.println(nomes.get(0)); // imprime certificação 


nomes .add(0, "java"); 
System.out.println(nomes.get(0)); // imprime java 
System.out.println(nomes.get(1)); // imprime certificação 


O mesmo acontece para o método remove: 


ArrayList<String> nomes = new ArrayList<String>0; 
nomes .add("java!'); 
nomes .add('"certificação!!); 


String removido = nomes.remove(0); // retorna java 
System.out.println(nomes.get(0)); // imprime certificação 
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E o método set, que serve para alterar o elemento em determinada 
posição: 


ArrayList<String> nomes = new ArrayList<String>(); 
nomes.add("java'); 
nomes.set (0, "certificação!"); 


System.out.printin(nomes.get(0)); // imprime certificação 
System.out.println(nomes.size(0)); // imprime 1 


Os métodos indexOfe last IndexOf retornam a primeira ou a última 
posição que possui o elemento desejado. Caso esse elemento não esteja na 
lista, ele retorna -1: 


ArrayList<String> nomes = new ArrayList<String>(); 
nomes.add("guilherme!'); 

nomes.add("mario"); 

nomes.add("paulo"); 

nomes.add("mauricio"); 

nomes.add("adriano"); 

nomes.add("alberto"); 

nomes.add("mario"); 


System. out .println (nomes .index0f ("guilherme")); // 0 
System. out .println (nomes .index0f("mario")); // 1 
System. out .println (nomes .index0f("joao")); // -1 
System. out .println (nomes .lastIndex0f("mario")); // 6 
System. out .println (nomes .lastIndex0f("joao")); // -1 


Iterator e o enhanced for 


A interface Iterator define uma maneira de percorrer coleções. Isso é 
necessário porque, em coleções diferentes de List, não possuímos métodos 
para pegar o enésimo elemento. Como, então, percorrer todos os elementos 
de uma coleção? 


e hasNext: retorna um booleano indicando se ainda há elementos a 
serem percorridos por esse iterador; 
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e next: pula para o próximo elemento, devolvendo-o; 


e remove: remove o elemento atual da coleção. 


O código que costuma aparecer para percorrer uma coleção é o seguinte: 


Collection<String> strings = new ArrayList<String>(); 
Iterator<String> iterator = strings.iterator(); 
while (iterator.hasNext()) 1 
String atual = iterator.next(); 
System.out.printin(atual); 


O enhanced-for também pode ser usado nesse caso: 


Collection<String> strings = new ArrayList<String>(); 
for (String atual : strings) { 
System.out.printlin(atual); 


O método equals em coleções 


A maioria absoluta das coleções usa o método equals na hora de buscar 
por elementos, como nos métodos contains e remove. Se você deseja ser 
capaz de remover ou buscar elementos, terá que provavelmente sobrescrever o 
método equals para refletir o conceito de igualdade em que está interessado, 
e não somente a igualdade de referência (implementação padrão do método). 

Cuidado ao tentar sobrescrever o método equals, se você escrevê-lo re- 
cebendo um tipo específico em vez de object, não o estará sobrescrevendo, 
eo ArrayList continuará invocando o código antigo, a implementação 
padrão de equals! 


ArrayList e referências 


Vale lembrar que Java sempre trabalha com referências para objetos, e não 
cria cópias de objetos cada vez que os atribuímos a uma variável ou referência: 


Cliente guilherme = new Cliente(); 
guilherme. setNome("guilherme") ; 
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ArrayList<Cliente> clientes = new ArrayList<Cliente>(); 
clientes.add(guilherme) ; 


System.out.printin(guilherme.getNome()); // guilherme 
System.out.printin(clientes.get (0) .getNome()); // guilherme 


guilherme. setNome ("Silveira"); 


System.out.println(guilherme.getNome()); // Silveira 
System.out.printin(clientes.get (0) .getNome()); // Silveira 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 

3 ArrayList<String> c = new ArrayList<String>(); 
4 c.add("a"); 

5 c.add("'c'); 

6 System.out.printin(c.remove("a")); 

7 } 

s } 


a) Não compila: erro ao declarar a ArrayList. 
b) Não compila: erro ao invocar remove. 

c) Compila e ao rodar imprime a. 

d) Compila e ao rodar imprime true. 


e) Compila e ao rodar imprime false. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı import java.util.ArrayList; 


2 class A { 

3 public static void main(String[] args) { 

4 ArrayList<String> c = new ArrayList<String>(); 
5 c.add("a"); 
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6 c.add("'c'); 

7 System.out.printin(c.remove("a")); 
8 } 

ə } 


a) Não compila: erro ao declarar a ArrayList. 


b) Não compila: erro ao invocar remove. 


c) Compila e ao rodar imprime a. 


d) Compila e ao rodar imprime true. 


e) Compila e ao rodar imprime false. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


import java.util.ArrayList; 
class A { 


n 


N 


w 


public static void main(String[] args) { 


4 ArrayList<String> c = new ArrayList<String>(); 
5 c.add("a'); 

6 c.add("a"); 

7 System.out.printin(c.remove("a")); 

8 System.out.printin(c.size()); 

9 } 

1 + 


a) Não compila: erro ao declarar a ArrayList. 


b) Não compila: erro ao invocar remove. 


c) Compila e ao rodar imprime 
d) Compila e ao rodar imprime 
e) Compila e ao rodar imprime 
f) Compila e ao rodar imprime 
g) Compila e ao rodar imprime 


h) Compila e ao rodar imprime 


aeo. 


Cru 


Cru 








Cru 


ael. 


eeo. 


eel 


aea. 


eemz. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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ı import java.util.ArrayList; 


2 class A { 

3 public static void main(String[] args) { 

4 ArrayList<String> list = new ArrayList<>(); 
5 list.add("a'");list.add("b''); 

6 list.add("a");list.add("c"); 

7 list.add("a");list.add("b"); 

8 list.add("a"); 

9 System.out.println(list.lastIndex0f("b')); 
10 } 

11 } 


a) Não compila. 

b) Compila e imprime -1. 
c) Compila e imprime o. 
d) Compila e imprime 1. 

e) Compila e imprime 2. 
f) Compila e imprime 3. 

g) Compila e imprime 4. 
h) Compila e imprime 5. 

i) Compila e imprime 6. 


j) Compila e imprime 7. 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı import java.util.ArrayList; 


2 class A { 

3 public static void main(String[] args) { 

4 ArrayList<String> 1 = new ArrayList<String>(); 
5 1l.add("a!'); 

6 1.add("b''); 

7 1.add(1,'"amor'"); 

8 1.add(3,'"baixinho!'); 

9 System.out.printin(1); 

10 String[] array = 1.toArray(); 
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u System.out.printin(array [2]); 


a) Não compila. 
b) Compila e imprime “amor”. 
c) Compila e imprime “b”. 
6) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı import java.util.ArrayList; 


2 class A { 

3 public static void main(String[] args) { 

4 ArrayList<String> a = new ArrayList<String>(); 
5 ArrayList<String> b = new ArrayList<String>(); 
6 ArrayList<String> c = new ArrayList<String>(); 
7 b.add("a");c.add("'c''); 

8 b.add("b");c.add("da"); 

9 a.addAlI(b); 

10 a.addAlI(c); 

u System.out.printin(a.get(0)); 

2 System.out.printin(a.get(3)); 

13 } 

14 > 


a) Não compila 
b) Compila e imprime ae d. 
c) Compila e imprime ce b. 


d) Compila e não sabemos a ordem em que os elementos serão impressos. 


7) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı import java.util.ArrayList; 


2 class A { 
3 public static void main(String[] args) { 
4 ArrayList<String> a = new ArrayList<String>(); 
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5 a.add("a!!, 0); 

6 a.add("b'!, 0); 

7 a.add('"'c'!, 0); 

8 a.add("'a', 0); 

9 System.out.printin(a.get(0)); 
10 System.out.println(a.get(1)); 
u System.out.println(a.get(2)); 
12 System.out.println(a.get(3)); 
13 } 

u > 


a) Não compila. 

b) Compila e imprime abca. 
c) Compila e imprime dcha. 
d) Compila e imprime adcb. 


e) Compila e imprime bcda. 


8) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı import java.util.x*; 


2 class À { 

3 public static void main(Stringl] args) 1 

4 ArrayList<String> a = new ArrayList<String>(); 
5 a.add(0,"b"); 

6 a:add(0,"a"); 

E for(Iterator<String> i=a.iterator();i.hasNext();i.next()) 1 
8 String element = i.next(); 

9 System.out.println(element); 

10 } 

11 } 

2 + 


a) Não compila. 
b) Compila e imprime a. 
c) Compila e imprime ae b. 


d) Compila e imprime be a. 
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e) Compila e imprime b. 


9) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı import java.util.ArrayList; 
2 class A { 


3 


4 


5 


6 


public static void main(Stringl] args) 1 


ArrayList<String> ss = new ArrayList<String>(); 
ss.add("a"); 
ss.add("b"); 
ss.add("c"); 
ss.add("d"); 


for(String s:ss){ 
if(s.equals("c")) s = "b"; 
else if(s.equals("b")) s= "c"; 
} 
for(String s:ss) System.out.println(s); 


a) Não compila, s é final por padrão. 


b) Compila e imprime a, c, b, d. 


c) Compila e imprime a, b, c, 


qd 
d) Compila e imprime a, c, c, d. 
qd 
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CAPÍTULO 7 


Usando laços 


7.1 CRIE E USE LAÇOS DO TIPO WHILE 


Outra maneira de controlar o fluxo de execução de um programa é definir 
que um determinado trecho de código deve executar várias vezes, como uma 
repetição ou um laço. 

Uma linguagem como o Java oferece alguns tipos de laços para o progra- 
mador escolher. O comando while é um deles. 


int i = 1; 
while (i < 10) { 
System.out.printin(i); 


i++; 


A sintaxe do while é a seguinte: 
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while (CONDICAO) 1 
// CODIGO 


Assim como no if, a condição de um bloco while deve ser um 
booleano. Da mesma maneira, se o bloco de código tiver apenas uma linha, 
podemos omitir as chaves: 


int i = 0; 
while( i < 10) 
System.out.println(i++); 


O corpo do while é executado repetidamente até que a condição se torne 
falsa. Em outras palavras, enquanto a condição for verdadeira. 

É necessário tomar cuidado para não escrever um while infinito, ou seja, 
um laço que não terminaria se fosse executado. 


int i = 1; 

//Quando fica false? 

while(i < 10){ 
System.out.println(i); 


Em casos em que é explícito que o loop será infinito, o compilador é es- 
perto e não deixa compilar caso tenha algum código após o laço: 


class A { 
int a() { 
while(true) { //nunca fica false 
System. out.println("Faz algo"); 


} 


return 1; // não compila, nunca chegará aqui 


Mesmo que a condição use uma variável, pode ocorrer um erro de com- 
pilação, caso a variável seja final: 


class A { 
int a) 4 
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final boolean RODANDO = true; 

while (RODANDO) 1 
System.out.println("'Faz algo"); 

} 


return 1; // não compila, nunca chegará aqui 


Agora, caso a variável não seja final, o compilador não tem como saber 
se o valor irá mudar ou não, por mais explícito que possa parecer, e o código 
compila normalmente: 


class A { 
int a() { 
boolean rodando = true; // não final 
while(rodando) { 
System. out.println("Faz algo"); 
} 
return 1; 
// compila, não tem como saber se o valor de rodando 
// vai mudar 


} 
} 
Caso um laço nunca seja executado, também teremos um erro de compi- 
lação: 


//unreachable statement, não compila. 
while(false) { //código aqui } 


//unreachable statement, não compila. 
while(1 > 2) { //código aqui 5 


Lembre-se que o compilador só consegue analisar operações com literais 
ou com constantes. No caso a seguir, o código compila, mesmo nunca sendo 
executado: 


int a il 
int b 2; 
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while(a > b){ //compila, mas nunca executa 
System.out.printin("OTI''); 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) { 
3 int a = 10; 

4 while(a>100) a++; 

5 System.out.printin(a); 

6 } 

7} 


a) Não compila pois nunca entra no loop. 
b) Compila e imprime 99. 

c) Compila e imprime 100. 

d) Compila e imprime 101. 


e) Compila e imprime outro valor. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 boolean rodar = true; 

4 while(rodar) 1 

5 System.out.printin(rodar); 

6 } 

7 System. out .println("Terminou"); 

8 } 

ə } 


a) Transformar a variávelem final faz o código compilar. 


b) Colocar uma linha dentro do laço que faz rodar = false faz o 
código compilar. 


c) O código compila e roda em loop infinito. 
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d) O código compila e roda, após algumas passagens pelo laço ele im- 
prime uma exception e para. 


7.2 CRIE E USE LAÇOS DO TIPO FOR, INCLUINDO O EN- 
HANCED FOR 


Observando um pouco os códigos que utilizam while, dá para perceber 
que eles são formados por quatro partes: inicialização, condição, comandos e 
atualização. 
int i = 1; // Inicialização 
while (i < 10) { // Condição 

System.out.printin(i); // Comandos 

i++; // Atualização 


A inicialização é importante para que o laço execute adequadamente. 
Mesmo com essa importância, a inicialização fica separada do while. 

A atualização é fundamental para que não aconteça um “loop infinito”. 
Porém, a sintaxe do while não a coloca em evidência. 

Há um outro laço que coloca em destaque a inicialização, a condição e a 
atualização. Esse laço é o for. 
for (int i = 1; i < 10; i++) { 

System.out.printin(i); 


O for tem três argumentos separados por ;. O primeiro é a inicialização, 
o segundo, a condição, e o terceiro, a atualização. 

A inicialização é executada somente uma vez no começo do for. A 
condição é verificada no começo de cada rodada (iteração). A atualização 
é executada no fim de cada iteração. 

Todos os três argumentos do for são opcionais. Desta forma, você pode- 
ria escrever o seguinte código: 


for(s:)1 
// CODIGO 
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O que acontece com esse laço? Para responder essa pergunta é necessário 
saber quais são os “valores default” colocados nos argumentos do for, 
quando não é colocado nada pelo programador. A inicialização e a atual- 
ização ficam realmente vazias. Agora, a condição recebe por padrão o valor 
true. Então, o código anterior depois de compilado fica assim: 


//loop infinito 
for (true) 
// CODIGO 


Nos exemplos anteriores, basicamente o que fizemos na inicialização foi 
declarar e inicializar apenas uma variável qualquer. Porém, é permitido 
declarar diversas variáveis de um mesmo tipo ou inicializar diversas variáveis. 

Na inicialização, não é permitido declarar variáveis de tipos diferentes. 
Mas é possível inicializar variáveis de tipos diferentes. Veja os exemplos: 


// Declarando três variáveis do tipo int e inicializando as três. 


// Repare que o "," separa as declarações e inicializações. 
for (int i=1,j=2,kK=3;; 

// CODIGO 
} 


// Declarando três variáveis de tipos diferentes 
int a; 

double b; 

boolean c; 


// Inicializando as três variáveis já declaradas 
for (a = 1, b= 2.0, c = true;;){ 
// CODIGO 


Na atualização, é possível fazer diversas atribuições separadas por ,. 
//a cada volta do laço, incrementamos o i e decrementamos o j 


for (int i=1,j=2;; i++,j--){ 
// código 
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Como já citamos anteriormente, não é possível inicializar variáveis de 
tipos diferentes: 


for (int i=1, long j=0; i< 10; i++){ // erro 
//código 


No campo de condição, podemos passar qualquer expressão que resulte 
em um boolean. São exatamente as mesmas regras do if e while. 

No campo de atualização, não podemos só usar os operadores de incre- 
mento, podemos executar qualquer trecho de código: 


for (int i = 0; i < 10; i += 3) { //somatório 
//código 


} 


for (int i = 0; i < 10; System.out.println(i++)) { // bizarro 
//código 


} 


Enhanced for 


Quando vamos percorrer uma coleção de objetos ou um array, podemos 
usar uma versão simplificada do for para percorrer essa coleção de maneira 
simplificada. Essa forma simplificada é chamada de enhanced for, ou foreach 


int[] numeros = {1,2,3,4,5,6}; 
for (int num : numeros) { //enhanced for 
System. out .println (num); 


A sintaxe é mais simples, temos agora 2 partes dentro da declaração do 


for: 


for (VARIAVEL : COLEÇÃO)( 
CODIGO 
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Nesse caso, declaramos uma variável que irá receber cada um dos mem- 
bros da coleção ou array que estamos percorrendo. O próprio for irá a cada 
iteração do laço atribuir o próximo elemento da lista à variável. Seria o equiv- 
alente a fazer o seguinte: 


int[] numeros = (1,2,3,4,5,6); 


for( int i=0; i < numeros.length; i++){ 
int num = numeros[il]; //declaração da variável e atribuição 
System.out.println (num) ; 


Se fosse uma collection, o código fica mais simples ainda se comparado 
com o for original: 


ArrayList<String> nomes = //lista com vários nomes 


//percorrendo a lista com o for simples 
for(Tterator<String> iterator = nomes.iterator(); 
iterator .hasNext();){ 
String nome = iterator .next(); 
System.out.println (nome); 


//percorrendo com o enhanced for 
for (String nome : nomes) { 
System.out.println (nome) ; 


Existem, porém, algumas limitações no enhanced for. Não podemos, por 
exemplo, modificar o conteúdo da coleção que estamos percorrendo usando 
a variável que declaramos: 


ArrayList<String> nomes = //lista com vários nomes 
//tentando remover nomes da lista 


for (String nome : nomes) { 
nome = null; 
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//o que imprime abaixo? 
for (String nome : nomes) { 
System. out .printin (nome); 


Ao executar esse código, você perceberá que a coleção não foi modificada, 
nenhum elemento mudou de valor para null. 

Outra limitação é que não há uma maneira natural de saber em qual iter- 
ação estamos, já que não existe nenhum contador. Para saber em qual linha 
estamos, precisaríamos de um contador externo. Também não é possível per- 
correr duas coleções ao mesmo tempo, já que não há um contador central- 
izado. Para todos esses casos, é recomendado usar o for simples. 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 for(;;) { 

4 System.out.println("a"); 

5 } 

6 System.out.println("b"); 

7 } 

s } 


a) Não compila. 

b) Compila e imprime a infinitamente. 

c) Compila e imprime b. 

d) Compila e imprime a, depois b, depois para. 


e) Compila, imprime a diversas vezes e depois dá um 





StackOverflowError 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 
2 public static void main(String[] args) { 
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3) 


1 
2 
3 
4 


5 


4) 


1 


2 
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for(;false;) 1 
System.out.printin("a"); 
break; 

} 

System.out.printin("b''); 


a) Não compila. 
b) Compila e imprime b. 


c) Compila, imprime ae b. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(String[] args) { 
for(int i=0, int j=1; i<10; i++, j++) System.out.println(i); 
} 
} 


a) Não compila. 

b) Compila e imprime o até 9. 

c) Compila e imprime o até 10. 
d) Compila e imprime 1 até 10. 


e) Compila e imprime 1 até 11. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(String[] args) { 
for(int i=0, j=1; i<10 ;i++, j++) System.out.println(i); 


a) Não compila. 


b) Compila e imprime o até 9. 
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5) 


6) 


c) Compila e imprime o até 10. 
d) Compila e imprime 1 até 10. 


e) Compila e imprime 1 até 11. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(Stringl] args) 1 
for(int i=0; i<10, false; i++) 1 
System.out.println('a'); 
} 
System.out.println('b'); 


a) Não compila. 
b) Compila e imprime 'a' e b. 


c) Compila e imprime ‘b. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(String[] args) { 
for(int i=0; i<2; i++, System.out.println(i)) { 
System.out.printin(i); 


a) Não compila. 

b) Compila e imprime 0 1 2. 

c) Compila e imprime 0 0 1 1 2 2. 
d) Compila e imprime 0 1 1 2 2. 


e) Compila e imprime 0 1 1 2. 
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7.3 CRIE E USO LAÇOS DO TIPO DO/WHILE 


Uma outra opção delaço whileseriao do .. while, que é bem parecido 
com o while. A grande diferença é que a condição é testada após o corpo 
do loop ser executado pelo menos uma vez: 


int i = 1; 

do { //executa ao menos 1 vez 
System.out.printin(i); 
i++; 


+ while (i < 10); // se der true, volta e executa novamente. 


A condição do do .. while só é verificada no final de cada iteração 
e não no começo, como no while. Repare que ao final do bloco do 
while existe um ponto e vírgula. Esse é um detalhe que passa desapercebido 
muitas vezes, mas que resulta em erro de compilação se omitido: 


int i = 1; 

do { 
System.out.printin(i); 
i++; 


} while (i < 10) // não compila, faltou o ; 


Assim como no while, caso tenhamos apenas uma linha, as chaves po- 
dem ser omitidas. Caso exista mais de uma linha dentro do do .. while 


e não existam chaves, teremos um erro de compilação: 


ant à = O; 
//compila normal 
do 
System.out.printin(i++); 
while(i<10); 


//erro, mais de uma linha dentro do do .. while 
do 
System.out.print('"o valor é: "); //erro 
System.out.printin(i++); 
while(i<10); 
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1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 boolean i = false; 

A do { 

5 System.out.printin(i); 

6 + while(i); 

7 E; 

8 + 


a) Não compila. 
b) Compila e imprime false. 


c) Compila e não imprime nada. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 if (args.length < 10) { 

3 do { 

5 if (args.length>2) return; 

6 + while(true); 

7 } 

8 System.out.println("Finalizou"); 

9 } 

10 + 


a) Não compila. 


b) Compila e entra em loop infinito caso seja passado zero, um ou dois 
argumentos. Não imprime nada caso 3 a 9 argumentos. Imprime 'Fi- 
nalizou” caso 10 ou mais argumentos. 


c) Compila e entra em loop infinito caso seja passado zero, um ou dois 
argumentos. Imprime “Finalizou” caso contrário. 


d) Compila e sempre entra em loop infinito. 
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3) 


1 
2 


3 


4) 


n 


5) 


1 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(String[] args) { 
int i = 0; 
do System.out.println(i); while(i++>10); 


a) Não compila. 

b) Compila e não imprime nada. 

c) Compila e imprime 0. 

d) Compila e imprime de 0 até 9. 
e) Compila e imprime de 0 até 10. 


f) Compila e não imprime nada. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(Stringl] args) 1 
int i = 0; 
do System.out.println(i) while(i++<10); 


a) Não compila. 
b) Compila e imprime de 0 até 9. 
c) Compila e imprime de 0 até 10. 


d) Compila e não imprime nada. 
Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
class A { 


public static void main(String[] args) { 
int i = 0; 
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4 do; while(i++<10); 
5 } 
6 5 


a) Não compila. 
b) Compila e entra em loop infinito. 


c) Compila e sai. 


7-4 COMPARE OS TIPOS DE LAÇOS 


Emborao for, whilee do .. while sejam todos estruturas que permi- 
tam executar loops, existem similaridades e diferenças entre essas construções 
que serão cobradas na prova. 


Comparando while e do .. while 


No caso do while e do while, ambos são muito similares, sendo a 
principal diferença o fato do do .. while ter a condição testada somente 
após executar o código de dentro do loop pelo menos uma vez. 


int i = 20; 


//imprime 20, já que só faz o teste após a execução do código 
do { 

System.out.printin(i); 

i++; 


+ while(i < 10); 
int j = 20; 


//não imprime nada, já que testa antes de executar o bloco 
while(j < 10)1 
System.out.printin(i); 


i++; 
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Comparando for e enhanced for 


Apesar de ser mais complexo, o for simples é mais poderoso que o 
enhanced for. Como enhanced for, não podemos: 


e Percorrer mais de uma coleção ao mesmo tempo; 
e Remover os elementos da coleção; 


e Inicializar um array. 


Caso desejemos fazer uma iteração de leitura, por todos os elementos 
da coleção, aí sim o enhanced for é a melhor opção. 


Comparando while e for 


Ambas estruturas de laço permitem executar as mesmas operações em 
nosso código, e são bem similares. Mas, apesar disso, existem situações em 
que o código ficará mais simples caso optemos por uma delas. 

Geralmente optamos por usar for quando sabemos a quantidade de 
vezes que queremos que o laço seja executado. Pode ser percorrer todos os 
elementos de uma coleção, (onde sabemos a quantidade de vezes que o loop 
será executado por saber o tamanho da coleção) ou simplesmente executar o 
laço uma quantidade fixa de vezes. 

Usamoso whileou do .. while quando não sabemos a quantidade 
de vezes em que o laço será executado, mas sabemos uma condição que, en- 
quanto for verdadeira, fará com que o laço seja repetido. 

O exemplo a seguir mostra um código no qual conhecemos a condição 
de parada, mas não faz sentido nenhum ter uma variável inicializada ou uma 
condição de incremento, então escolhemos um while: 


while(conta.getSaldo() > 0) 1 
conta. saca(1000); 


Note que, caso queira contar quantas vezes foi sacado, faria sentido usar 
um for: 
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int saques; 

for(saques = 0; conta.getSaldo() > 0; saquest+) { 
conta. saca(1000); 

} 


System.out.printin("Saquei " + saques + " vezes"); 


1) Qual o laço mais simples de ser usado quando desejamos iterar por duas 
coleções ao mesmo tempo? 


a) for 
b) while 
c) enhanced for 


d) do... while 


2) Qual o melhor laço a ser usado para, dependendo do valor de um ele- 
mento, removê-lo de nossa lista? 


a) for 


b) enhanced for 


3) Para todos os números entre i e 100 devo imprimir algo, sendo que, 
mesmo que i seja maior que 100, devo imprimir algo pelo menos uma 
vez. Qual laço devo usar? 


a) enhanced for 
b) do... while 


c) while 


4) Qual laço a ser usado caso queira executar um código eternamente? 


a) enhanced for 
b) for 


c) for ou while 
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d) for ou while ou do...while 


e) while ou do...while 


5) Qual laço deve ser usado para inicializar os valores de um array? 


a) enhanced for 


b) for 


7.5 USE BREAK E CONTINUE 


Em qualquer estrutura de laço podemos aplicar os controladores break e con- 
tinue. O break serve para parar o laço totalmente. Já o continue inter- 
rompe apenas a iteração atual. Vamos ver alguns exemplos: 


int i = 1; 
while (i < 10) { 
i++; 
if (i == 5) 
break; // sai do while com i valendo 5 
System.out.println(i); 
} 


System. out.println("Fim"); 


Ao executar o break, a execução do while para completamente. Temos 
a seguinte saída: 


2 
3 
4 
Fim 


Vamos comparar com o continue: 
int i = 1; 


while (i < 10) { 


i++; 


if (i == 5) 
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continue; // vai para a condição com o i valendo 5 
System.out.printin(i); 


Neste caso, iremos parar a execução da iteração apenas quando o valor da 
variável for igual a 5. Ao encontrar um continue, o código volta ao início 
da iteração, ao ponto do loop. Nossa saída agora é a seguinte: 


Isto é, o break quebra o laço atual, enquanto o continue vai para a 
próxima iteração do laço. 

Tome cuidado, pois um laço que tenha um while infinito do tipo true 
e que contenha um break é compilável, já que o compilador não sabe se o 
código poderá parar, possivelmente sim: 


while(true) { 
if(1==2) break; 
System.out.printin("em loop infinito compilável"); 


Os controladores de laços, break e continue podem ser aplicados no 
for. O break se comporta da mesma maneira que no while eno do 
while, parar o laço por completo. Já o continue faz com que a iteração 
atual seja abortada, executando em seguida a parte de atualização do for, e 
em seguida a de condição. Vamos ver o exemplo a seguir: 


for (int i = 1; i < 10; i++) { 
if (i = 8) { 
break; // sai do for sem executar mais nada do laço. 


191 


7.5. Use break e continue Casa do Código 





if (i == 5) { 
// pula para a atualização sem executar o resto do corpo. 


continue; 

} 

System.out.println(i); 
} 

A saída desse código é : 

1 
2 
3 
4 
6 
T 


Rótulos em laços (labeled loops) 


Às vezes, encontramos a necessidade de “encaixar” um laço dentro de 
outro. Por exemplo, um for dentro de um while ou de outro for. Nesses 
casos, pode ser preciso manipular melhor a execução dos laços encaixados 
com os controladores de laços, break e continue. 


for (int i = 1; i < 10; i++) { //laço externo 
for (int j = 1; j < 10; j++) { // laço interno 
if (ix j == 25) { 
break; // qual for será quebrado? 


Quando utilizamos o break ouo continue em laços encaixados, eles 
são aplicados no laço mais próximo. Por exemplo, nesse código, o break irá 
cC » . . . cc » . 

quebrar” o for mais interno. Se fosse preciso “quebrar” o for mais externo, 
como faríamos? 


Labeled statements 


Podemos adicionar labels (rótulos) a algumas estruturas de código, e usá- 
los posteriormente para referenciarmos essas estruturas. Para declarar um 
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label usamos um nome qualquer (mesma regra de nomes de variáveis etc.) 
seguido de dois pontos ( :). Por exemplo, podemos dar um label para um 
for como o que segue: 


externo: //label 
for(int i=0; i<10;i++){ 
//código 


Podemos usar esses labels para referenciar para qual loop queremos que 
o break ouo continue seja executado: 


externo: for (int i = 1; i < 10; i++) { 
interno: for (int j = 1; j < 10; j++) { 
if (i * j == 25) { 
break externo; // quebrando o for externo 
F 
if (i *» j == 16) { 
continue interno; // pulando um iteração do for interno 





LABEL HTTP 


O código a seguir imprime os valores de 1 a 10. Mas como ele compila 
sendo que temos uma URI logo antes do laço for? 


http://www. caelum. com.br 
for (int i = 1; i <= 10; i++) { 
System.out.println(i); 











Um rótulo ou label pode estar presente antes de um statement qualquer, 
mas só podemos utilizar um statement de break ou continue caso o rótulo 
esteja referenciando um for, while ou switch: 
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void rotuloEmQualquerLugar O { 
rotulo: System.out.printin("oi"); 


void rotuloEmQualquerLugarComBreakNaoCompila() 1 
rotulo: System.out.printin("oi"); 
if (1<10) continue rotulo; // erro de compilação 


Cuidado, mesmo dentro de um for ou similar, o continue eo break 
só funcionarão se forem relativos a um label dentro do qual estão, e do tipo 
for, do.. while, switchou while. Valelembrar que switch só aceita 


break. 


void rotuloEmQualquerLugarComBreakNaoCompila() 1 
rotulo: System.out.printin("oi"); 
for(int i=0;i<10;i++) { 
break rotulo; // não compila 


} 
void rotuloEm0utroLaco() { 
rotulo: 
for(int i=0;i<10;i++) { 
System.out.println("oi"); 
} 
for(int i=0;i<10;i++) { 
break rotulo; // não compila 


Rótulos podem ser repetidos desde que não exista conflito de escopo: 


void rotulosRepetidos() { 
rotulo: for (int i = 0; i < 10; i++) { 
break rotulo; 


} 
rotulo: for (int i = 0; i < 10; i++) { 
break rotulo; 
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E; 
void rotulosRepetidosNestedNaoCompila() f 
rotulo: for (int i = 0; i < 10; i++) { 
rotulo: for (int j = 0 j < 10; J+) {£ 
break rotulo; 


Não há conflito de nome entre rótulos e variáveis, pois seu uso é bem 
distinto. O compilador sabe se você está referenciando um rótulo ou uma 
variável: 


class A { 
int rotulo = 15; 
void rotulosENomesDeVariaveisNaoConflitam() { 
rotulo: for (int i = 0; i < 10; i++) { 
int rotulo = 10; 
break rotulo; 


Um mesmo statement pode ter dois labels: 


void rotulosNoMesmoStatement () { 
primeiro: segundo: for (int i = 0; i < 10; i++) { 
System.out.printin(i); 


Tome bastante cuidado com breaks e continues que são de switch 
mas parecem ser de fors: 


class TestaLacos { 
public static void main(Stringl[] args) 1 


for(int i = 0; i < 4; i++) { 


System.out.printin("Estou antes do switch"); 


mario: 
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guilherme: switch(i) 1 
case O: 
case 1: 
System.out.printin("Caso " + i); 
for(int j = 0; j < 3; j++) 4 
System. out.println(j); 
if(j==1) break mario; 
} 
case 2: 
System.out.printin("Estou em i = " + i); 
continue; 
case 3: 
System.out.printin("Cheguei no 3"); 
break; 
default: 
System.out.printin("Estranho...''); 
break; 
} 


System.out.printin("Estou apos o switch"); 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 fora: for(int a=0;a<30;a++) 

4 for(int b=0;b<1;b++) 

5 if (atb==25) continue fora; 

6 else if(a+b==20) break fora; 

7 if (a==0) break fora; 

8 else System.out.printin(a); 

9 } 

10 + 


a) Não compila. 


b) Compila e imprime 1 até 29 
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c) Compila e não imprime nada 
d) Compila e imprime 1 até 19, 21 até 24, 26 até 29 
e) Compila e imprime 1 até 24, 26 até 29 


f) Compila e imprime 1 até 19 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class À É 

2 public static void main(Stringl] args) 1 
3 fora: for(int a=0;a<30;a++) 

4 for(int b=0;b<1;b++) 

5 if(a+b==25) continue fora; 

6 else if(atb==20) break; 

7 else System.out.println(a); 

8 } 

9 5 


a) Não compila. 

b) Compila e imprime o até 29. 

c) Compila e não imprime nada. 

d) Compila e imprime o até 19, 21 até 24, 26 até 29. 
e) Compila e imprime o até 24, 26 até 29. 


f) Compila e imprime o até 19. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A É 

2 public static void main(String[] args) { 

3 int a = args.length; 

4 int i = 0; 

r switch(a) { 

6 case 0: 

7 case 1: 

8 for(i=0;i<15;i++, System.out.printin(i)) 
9 if(i==5) continue; 
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if(i==15) break; 
case 2: 
System.out.printin('"2"); 
} 
System. out.println("fim"); 


Não compila. 

Compila e ao rodar com o argumentos imprime o até 14, 2, fim. 
Compila e ao rodar com o argumentos imprime 1 até 15, 2, fim. 
Compila e ao rodar com o argumentos imprime o até 4, 6 até 14, 2, fim. 
Compila e ao rodar com o argumentos imprime 1 até 4, 6 até 15, fim. 
Compila e ao rodar com o argumentos imprime o até 4, 6 até 9, 2, fim. 
Compila e ao rodar com o argumentos imprime 1 até 4, 6 até 9, fim. 
Compila e ao rodar com o argumentos imprime 1 até 4, 6 até 15, 2, fim. 
Compila e ao rodar com o argumentos imprime 1 até 15, 2, fim. 


Compila e ao rodar com o argumentos imprime 1 até 15, fim. 


CAPÍTULO 8 


Trabalhando com métodos e 
encapsulamento 


8.1 CRIE MÉTODOS COM ARGUMENTOS E VALORES DE 
RETORNO 


Classes, enums e interfaces podem ter métodos definidos em seus corpos. 
Todo método tem uma assinatura (também chamada de interface) e um 
corpo (somente no caso de métodos não abstratos). 


A assinatura do método sempre tem: 


* um nome seguindo as regras de identificadores; 


* um tipo de retorno; 
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* um conjunto de parâmetros (pode ser vazio), cada um com seu nome 
e seu tipo; 


e um modificador de visibilidade (nem que seja implícito, package- 
private). 


E, ainda na assinatura, podemos ter: 


e final em caso de herança, o método não pode ser sobrescrito nas 
classes filhas; 


e abstract obriga as classes filhas a implementarem o método. O 
método abstrato não pode ter corpo definido; 


e static atributos acessados direto na classe, sem instâncias; 
e synchronized lock da instância; 


* native não cai nesta prova. Permite a implementação do método em 
código nativo (JNT); 


e strictfp não cai nesta prova. Ativa o modo de portabilidade 
matemática para contas de ponto flutuante. 











e throws <EXCEPTIONS> após a lista de parâmetros, podemos indicar 





quantas exceptions quisermos para O throws. 


A ordem dos elementos na assinatura dos métodos é sempre a seguinte, 
sendo que os modificadores podem aparecer em qualquer ordem: 
<MODIFICADORES> <TIPO. RETORNO> <NOME> (<PARÂMETROS>) 
<THROWS. EXCEPTIONS> 

















T 














Parâmetros 


Em Java, usamos parâmetros em métodos e construtores. Definimos uma 
lista de parâmetros sempre declarando seus tipos e nomes e separando por 
vírgula: 
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class Param { 
void teste(int a, int b) É 


// chamada 
p.teste(l, 2); 


A declaração das variáveis é feita na declaração dos métodos. A inicial- 
ização dos valores é feita por quem chama o método. (Note que, em Java, não 
é possível ter valores default para parâmetros e todos são obrigatórios, não 
podemos deixar de passar nenhum). 

O único modificador possível de ser marcado em parâmetros é final, 
para indicar que aquele parâmetro não pode ter seu valor modificado depois 
da chamada do método (considerado boa prática): 


class Param { 
void teste (final int a) { 


a = 10; // não compila 


Promoção em parâmetros 


Temos que saber que nossos parâmetros também estão sujeitos à pro- 
moção de primitivos e ao polimorfismo. Por exemplo, a classe a seguir ilustra 
as duas situações: 


class Param { 
void primitivo (double d) { 


void referencia (Object o) { 
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O primeiro método espera um double. Mas se chamarmos passando 
um int,um float ou qualquer outro tipo compatível, este será promovido 
a double e a chamada funciona: 


Param p = new Param(); 
p.primitivo(10); 
p.primitivo(iOL); 
p.primitivo(10F); 
p.primitivo((short) 10); 
p.primitivo((byte) 10); 
p-.primitivo('Z'); 


A mesma coisa ocorre com o método que recebe Object: podemos pas- 
sar qualquer um que é um Object, ou seja, qualquer objeto: 


Param p = new Param(); 
p.referencia(new Carro()); 
p-.referencia(new Moto()); 


Retornando valores 


Todo método pode retornar um valor ou ser definido como void, 
quando não devolve nada: 


class A { 
int numero() { 
return 5; 


} 
void nada() { 
return; 


No caso de métodos de tipo de retorno void (nada), podemos omitir a 
última instrução: 


class AÍ 
void nada() { 
// return; // pois esta linha é opcional 


202 


Casa do Código Capítulo 8. Trabalhando com métodos e encapsulamento 





Um método desse tipo também pode ter um retorno antecipado: 


class A { 
void nada(int i) { 
if(i >= 0) return; 
System. out .println("negativo"); 


} 
} 
Não podemos ter nenhum código que seria executado após um retorno: 
class A { 
void nada(int i) { 
if(i >= 0) { 
return; 
// não compila, pois nunca chegará aqui 
System.out.println('era positivo ou zero"); 
} 
System. out .println("negativo"); 
} 
} 


Todo método que possui um tipo de retorno definido (isto é, diferente 





de void), deve retornar algo ou jogar uma Exception em cada um dos 
caminhos de saída possíveis do método, caso contrário o código não compila: 


String metodo(int a) { 
if(a > 0) { 
return "positivo"; 
} else if(a < 0) + 
return "negativo"; 


} 


//não compila, o que acontece se não for nem if nem else if? 


Lembre-se que isso é feito pelo compilador, então ele não sabe os valores 
da variável a e se todos os casos foram cobertos: 


String metodo(int a) 1 
if(a > 0) 1 
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return "positivo"; 
} else if(a <= 0) { 

return "negativo ou zero"; 
//não compila, o que acontece se não for nem if nem else if? 
//o compilador não consegue analisar os dois casos 


Podemos jogar uma exception ou colocar um return: 


String metodo(int a) 1 
if(a > 0) 1 
return "positivo"; 
} else if(a < 0) 1 
return "negativo"; 


J 


return "zero"; 


String metodo2(int a) { 
if(a > 0) { 
return "positivo"; 
} else if(a < 0) 1 
return "negativo"; 


} 


throw new RuntimeException("não quero zero!"); 


Métodos que não retornam nada não podem ter seu resultado atribuído 
a uma variável: 


void metodo() { 
System.out.println("oi"); 


} 

void metodo2() { 
// não compila, o método acima não retorna nada 
int i = metodo(); 


Pelo outro lado, mesmo que um método retorne algo, seu retorno pode 
ser ignorado: 
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int metodo() 1 
System.out.printin("oi'); 
return 5; 
} 
void metodo2() { 
int i = metodo(); //i=5 
// chamei novamente e não retornei nada, sem problemas 
metodo(); 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 x(args.length); 

4 } 

5 static int x(final int 1) É 

6 for(int i=0;i<100;i++) { 

z switch(i) { 

8 case 1: 

9 System.out.printin(1); 
10 if(1l==i) return; 

u case 0: 

12 System.out.printin(0); 
5 E; 

14 } 

15 System.out.println("Fim"); 

16 return -1; 

17 } 

18 + 


a) Não compila. 

b) Compila e ao rodar com cinco parâmetros, imprime o, 5 e Fim. 

c) Compila e ao rodar com cinco parâmetros, imprime o, 5, -1 e Fim. 
d) Compila e ao rodar com cinco parâmetros, imprime o e 5. 

e) Compila e ao rodar com cinco parâmetros, imprime o, 5 e -1. 


f) Compila e ao rodar com cinco parâmetros, imprime o, 5, o e Fim. 
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g) Compila e ao rodar com cinco parâmetros, imprime o, 5, 0, -1 e Fim. 
h) Compila e ao rodar com cinco parâmetros, imprime o e 5. 


i) Compila e ao rodar com cinco parâmetros, imprime 0,5,0 e 1. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 
3 x(args. length); 

4 } 

5 static int x(final int 1) 1 

6 for(int i=0;i<100;i++) { 

3 switch(i) { 

8 case 1: 

9 System.out.printin(1); 
10 if (1==i) return 3; 

u case 0: 

12 System.out.println(0); 
13 } 

14 } 

15 System. out.println("Fim"); 

16 return -1; 

17 } 

18 } 


a) Não compila. 

b) Compila e ao rodar com cinco parâmetros, imprime o, 5 e Fim. 

c) Compila e ao rodar com cinco parâmetros, imprime o, 5, -1 e Fim. 
d) Compila e ao rodar com cinco parâmetros, imprime o e 5. 

e) Compila e ao rodar com cinco parâmetros, imprime o, 5 e -1. 

f) Compila e ao rodar com cinco parâmetros, imprime o, 5, o e Fim. 

g) Compila e ao rodar com cinco parâmetros, imprime o, 5, 0, -1 e Fim. 
h) Compila e ao rodar com cinco parâmetros, imprime o e 5. 


i) Compila e ao rodar com cinco parâmetros, imprime 0,5,0 e 1. 
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3) 


1 
2 
3 
4 


5 


4) 


1 
2 
3 


4 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(Stringl] args) 1 
System.out.println(a(args.length)); 


} 

static int a(int 1) { 
if (1<10) return b(1); 
else return c(); 


} 

static int b(int 1) { 
if (1<10) return b(1); 
else return c(); 


$ 
static long c() { 
return 3; 


a) Não compila: erro ao invocar o método b. 

b) Não compila: erro ao invocar o método c. 

c) Não compila por um motivo não listado. 

d) Compila e, ao chamar com 15 argumentos, imprime 3. 


e) Compila e, ao chamar com 15 argumentos, entra em loop infinito. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 
public static void main(String[] args) { 
System.out.printin(a(args.length) [0]); 
} 


static int, int alint 1) { 
if (1==0) return {0, 1}; 
else return {1, O}; 


a) Não compila. 
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b) Ao invocar com nenhum parâmetro, imprime o. 


c) Ao invocar com 5 parâmetros, imprime o. 


8.2 APLIQUE A PALAVRA CHAVE STATIC A MÉTODOS E 
CAMPOS 


O modificador estático diz que determinado atributo ou método pertence à 
classe, e não a cada objeto. Com isso, você não precisa de uma instância para 


acessar o atributo, basta o nome da classe. 


public class Carro { 
public static int totalDeCarros; 


E depois, para acessar: 


Carro .totalDeCarros = 5; 


Um método estático é um método da classe, podendo ser chamado sem 
uma instância: 


public class Carrof 
private static int totalDeCarros; 


public static int getTotalDeCarros() [ 
return totalDeCarros; 


int i = Carro.getTotalDeCarros(); 


O que não podemos fazer é usar um método/atributo de instância de den- 
tro de um método estático: 


public class Carrof 
private static int totalDeCarros; 
private int peso; 
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public static int getPeso() { 
return peso; 


Esse código não compila, pois peso é um atributo de instância. Se alguém 
chamar esse método, que valor ele retornaria, já que não estamos trabalhando 
com nenhuma instância de carro em específico? 

Repare que a variável estática pode acessar um método estático, e esse 
método acessar algo ainda não definido e ter um resultado inesperado à 
primeira vista: 


static int b = getMetodo(); 
public static int getMetodo() { 
return a; 


} 


static int a = 15; 


O valor de b será o, e não 15, uma vez que a variável a ainda não foi inicial- 
izada e possui seu valor padrão quando da execução do método getMetodo. 

Outro caso interessante é que uma variável estática pode acessar outra 
estática, desde que a outra tenha um valor atribuído antes da definição da 
atual: 


static int inicial 10; 


static int segunda = inicial + 5; // compila 


static int outra; 
static void inicializa() { 
outra = 10; 
} 
static int naoCompila = outra + 1; 
// não compila, o método inicializa é ignorável 


Um detalhe importante é que membros estáticos podem ser acessados 
através de instâncias da classe (além do acesso direto pelo nome da classe). 


Carro c = new Carro(); 
int i = c.getTotalDeCarros(); 
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Cuidado com essa sintaxe, que pode levar a acreditar que é um método de 
instância. É uma sintaxe estranha mas que compila e acessa o método estático 
normalmente. 

Além disso, esteja atento pois, caso uma classe possua um método es- 
tático, ela não pode possuir outro método não estático com assinatura que a 
sobrescreveria (mesmo que em classe mãe/filha): 


class A { 
static void a() { // não compila 
+ 
void a() { // não compila 
} 
} 
class B { 
static void a() { 
} 
} 


class C extends B { 
void a() { // não compila 


} 


Outro ponto importante a tomar nota é que o binding do método é feito 
em compilação, portanto, o método invocado não é detectado em tempo de 
execução. Leve em consideração: 


class A { 
static void metodo() { 


System.out.println("a"); 


class B extends A { 
static void metodo() { 


System.out.println("b"); 
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Caso o tipo referenciado de uma variável seja A em tempo de compilação, 
o método será o da classe A. Se for referenciado como B, será o método da 
classe B: 


A a= new A(); 
a.metodo(); // a 


B b= new B(); 
b.metodo(); // b 


A a2 = b; 
a2.metodo(); // a 
} 
} 
A definição de uma variável estática pode invocar métodos e variáveis 
estáticas: 
class A { 
static int idade = calculaldade(); 
static int calculaľdade() { 
return 18; 
} 
} 


A palavra-chave static pode ser aplicada a classes aninhadas, mas este 
tópico não é cobrado nesta primeira certificação. 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 x0; 

4 } 

5 static xO 1 

6 System.out.println("x'); 

7 yO; 

8 } 

f static y() { 
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10 System. out.println("y"); 


a) Não compila. 
b) Imprime x, y. 
c) Imprime y, x. 
d) Imprime x. 


e) Imprime y. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 x(); 

4 } 

5 static void x() | 

6 System. out.println("x"); 
7 yO; 

8 } 

3 static void y() { 

10 System. out.println("y"); 
nu } 

2 + 


a) Não compila. 
b) Imprime x, y. 
c) Imprime y, x. 
d) Imprime x. 


e) Imprime y. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { 
2 void y() | 
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3 this.z(); 

4 } 

E static void z() | 

6 System.out.println(''z'); 
7 F} 

8 + 

ə class A { 

10 public static void main(String[] args) { 
n new A().x(); 

12 } 

i3 static void x() { 

14 new B().y0; 

15 F 

16 } 


a) Não compila ao tentar invocar y. 
b) Não compila ao tentar invocar z. 
c) Não compila ao tentar invocar x. 


d) Compila e imprime z. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B{ 

2 static void x() { 

3 System.out.println("x"); 
4 } 

5 static void y() { 

6 System.out.println("y"); 
7 } 

8 + 

ə class A extends B { 

10 public static void main(String[] args) { 
u this.x0O; 

12 A.yO; 

13 } 

4 } 


a) Não compila, erro ao invocar x. 
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b) Imprime x, y. 


c) Não compila, erro ao invocar y. 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class BÍ 

2 static void x() { 

3 System.out.printin("x''); 
4 } 

; static void y() { 

6 System. out.println("y"); 
7 } 

eF 

ə class A extends B { 

10 public static void main(String[] args) { 
u xC); 

a A.yO; 

13 } 

14 > 


a) Não compila, erro ao invocar x. 
b) Imprime x, y. 


c) Não compila, erro ao invocar y. 


8.3 CRIE MÉTODOS SOBRECARREGADOS 


Um método pode ter o mesmo nome que outro, desde que a chamada não 
fique ambígua: os argumentos que são recebidos têm de ser obrigatoriamente 
diferentes, seja em quantidade ou em tipos. 


class Teste { 
public void metodo(int i) { 
} 


protected void metodo(double x) { 
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Já o código a seguir não compila: 


class Teste { 
public int metodo() 4{} 
protected double metodo() {} 


Nesse exemplo, temos ambiguidade porque o tipo de retorno não é sufi- 
ciente para distinguir os métodos durante a chamada. 

O Java decide qual das assinaturas de método sobrecarregado (over- 
loaded) será utilizada em tempo de compilação. 

Métodos sobrecarregados podem ter ou não um retorno diferente e uma 
visibilidade diferente. Mas eles não podem ter exatamente os mesmos tipos e 
quantidade de parâmetros. Nesse caso, seria uma sobrescrita de método. 

No caso de sobrecarga com tipos que possuem polimorfismo, como em 
Object ou String, o compilador sempre invoca o método com o tipo mais 
específico (menos genérico): 


public class Teste { 
void metodo(Object o) { 
System.out.println("object"); 
} 
void metodo(String s) { 
System. out.println("string"); 


public static void main(String[] args) { 
new Teste().metodo("string"); // imprime string 


Se quisermos forçar a invocação ao método mais genérico, devemos fazer 
o casting forçado: 


public class Teste { 
void metodo(Object o) { 
System.out.println("object"); 
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void metodo(String s) { 
System.out.println("string'); 


public static void main(Stringl] args) 1 
new Teste() .metodo((Object)"'string'"); // imprime object 


Um exemplo clássico é a troca de ordem, que é vista como sobrecarga, 
afinal são dois métodos totalmente distintos: 


void metodo(String i, double x) 1 


} 
void metodo(double x, String i) { 


} 


Porém, apesar de compiláveis, às vezes o compilador não sabe qual 
método deverá chamar. No caso a seguir, os números 2 e 3 podem ser consid- 
erados tanto int quanto double, portanto, o compilador fica perdido em 
qual dos dois métodos invocar, e decide não compilar: 


public class Teste { 
void metodo(int i, double x) { 
} 


void metodo(double x, int i) É 


} 


public static void main(String[] args) { 
new Teste() .metodo(2, 3); 


Isso também ocorre com referências, que é diferente do caso com tipo 
mais específico. Aqui não há tipo mais específico, pois onde um é mais es- 


pecífico, o outro é mais genérico: 


public class Xpto { 
void metodo(Object o, String s) { 
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System. out .printin("object'"); 
} 
void metodo(String s, Object o) { 
System.out.println("string"); 


public static void main(Stringl[] args) { 
new Xpto() .metodo("string", "string"); 


Diferente do caso em que o segundo método é mais específico: 


class Xpto2 { 
void metodo (Object o, Object 02) 1 
System.out.println("object"); 
} 
void metodo(String s, String s2) { 
System. out.println("string"); 


public static void main(String[] args) { 
new Xpto2().metodo("string", "string"); // imprime string 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 int x = b(15); 

4 System.out.printin(x); 

5 System.out.printIn(15); 

6 System.out.println(15.0); 

7 } 

8 static int blint i) { return i; 5 

9 static double b(int i) { return i; + 

10 + 
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a) Não compila. 
b) Compila e imprime 15, 15, 15. 
c) Compila e imprime 15, 15, 15.0. 


d) Compila e imprime 15, 15.0, 15.0. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
š int x = b(15); 

4 System.out.println(x); 

5 System.out.println(15); 

6 System.out.println(15.0); 

7 } 

8 static int b(int i) { return i; 5 

9 static double b(double i) { return i; 5 
10 + 


a) Não compila. 
b) Compila e imprime 15, 15, 15. 
c) Compila e imprime 15, 15, 15.0. 


d) Compila e imprime 15, 15.0, 15.0. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 System. out .println("[]"); 

4 } 

5 public static void main(String... args) É 
6 System.out.println("...'"); 

7 } 

s } 


a) Não compila. 


» 


b) Compila e imprime “[]”. 
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4) 


N 


w 


5) 


N 


w 


«>» 


c) Compila e imprime “.” 


d) Compila e dá exception. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B{} 

class C{} 

class D extends B(J 

class A { 
int a(D d) { return 1; 5 
int a(C c) { return 2; + 
int a(B b) { return 3; 5 
int a(A a) { return 4; + 
public static void main(Stringl] args) 1 

System.out .printin(a(new D())); 


} 


a) Não compila. 

b) Compila e imprime 1. 
c) Compila e imprime 2. 
d) Compila e imprime 3. 


e) Compila e imprime 4. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B{} 
class C{} 
class D extends B{} 
class A { 
int a(D d) { return 1; } 
static int a(C c) { return 2; } 
static int a(B b) { return 3; 5 
static int a(A a) { return 4; 5 
public static void main(Stringl] args) 1 
System.out.printin(a(new D())); 
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6) 


nm 


N 


w 


7) 


nm 


a 


3 


4 


a) Não compila. 

b) Compila e imprime 1. 
c) Compila e imprime 2. 
d) Compila e imprime 3. 


e) Compila e imprime 4. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B{} 

class C{} 

class D extends B{} 

class A { 
static int a(D d) É 
static int a(C c) É 
static int a(B b) É 
static int a(A a) É 


public static void main(Stringl] args) 1 
System.out.printin(a(new D())); 


a) Não compila. 

b) Compila e imprime 1. 
c) Compila e imprime 2. 
d) Compila e imprime 3. 


e) Compila e imprime 4. 


return 
return 
return 
return 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B{} 
class C{} 
class D extends B(J 
class A { 


static int a(D d, Bb) { return 1; 5 
static int a(C c, Cc) { return 2; 5 
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static int a(B b, Bb) { return 3; + 

static int a(Ã a, Aa) { return 4; + 

public static void main(Stringl] args) 1 
System.out.printin(a(new D(), new D())); 


a) Não compila. 


b) Compila e imprime 1. 


c) Compila e imprime 2. 


d) Compila e imprime 3. 


e) Compila e imprime 4. 


8) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


1 


2 


class B{} 
class C{} 


3 class D extends B{} 


4 


5 


6 


class A { 
static 
static 
static 
static 
public 


int a(D d, B b2) { return 1; 5 
int a(C c, € c2) { return 2; 5 
int a(B b, B b2) { return 3; 5 
int a(A a, A a2) { return 4; 5 
static void main(Stringl[] args) { 


System.out .printin(a(new D(), new D())); 


a) Não compila. 


b) Compila e imprime 1. 


c) Compila e imprime 2. 


d) Compila e imprime 3. 


e) Compila e imprime 4. 
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8.4 DIFERENCIA ENTRE O CONSTRUTOR PADRÃO E 
CONSTRUTORES DEFINIDOS PELO USUÁRIO 


Quando não escrevemos um construtor na nossa classe, o compilador nos 
dá um construtor padrão. Esse construtor, chamado de default não recebe 
argumentos, tem a mesma visibilidade da classe e tem a chamada a super (). 


A classe a seguir: 


class A { 
+ 


... na verdade, acaba sendo: 


class A { 
AO | 


super (); 


Caso você adicione um construtor qualquer, o construtor default deixa 
de existir: 


class A {} 
class B { 
B(String s) {} 
} 
class Teste { 
public static void main(String[] args) { 
new A(); // construtor padrão, compila 
new B(); // não existe mais construtor padrão 
new B("CDC"); // construtor existente 


Dentro de um construtor você pode acessar e atribuir valores aos atribu- 
tos, suas variáveis membro: 


class Teste { 


ime DS 
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Teste() 1 
i = 15; // agora i vale 15 
System.out.printin(i); // 15 


public static void main(Stringl[] args) { 
new Teste(); 


Os valores inicializados com a declaração das variáveis são inicializados 
antes do construtor, justamente por isso o valor inicial de i éo, o valor padrão 
de uma variável int membro: 


class Teste { 
int i; 
Teste() { 
System.out.println(i); // vale O por padrão 
i = 15; // agora à vale 15 
System.out.println(i); // 15 


public static void main(Stringl[] args) 1 
new Teste(); 


Vale lembrar que variáveis membro são inicializadas automaticamente 
para: numéricas 0, boolean false, referências null. 

Cuidado ao acessar métodos cujas variáveis ainda não foram inicializadas 
no construtor. O exemplo a seguir mostra um caso em que o método de ini- 
cialização é invocado antes de setar o valor da variável no construtor, o que 





causa um NullPointerException 





class A { 
int i = 15; 
String nome; 


int tamanho = tamanhoDoNome() ; 
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A(String nome) 1 
this.nome = nome; 


int tamanhoDoNome() 1 
return nome.length(); 


AO 1 
F 


Mesmo que inicializemos a variável fora do construtor, após a chamada 


do método pode ocorrer um erro, como no caso a seguir, de um outro 





NullPointerException 





class A { 


int i = 15; 

String nome; 

int tamanho = tamanhoDoSobrenome(); 
String sobrenome = "Silveira"; 


A(String nome) { 


this.nome = nome; 


int tamanhoDoSobrenome() { 
return sobrenome.length(); 


AO { 
} 


Mudar a ordem da declaração das variáveis resolve o problema, uma vez 


que o método é agora invocado após a inicialização da variável sobrenome: 
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class A { 


int i = 15; 

String nome; 

String sobrenome = "Silveira"; 

int tamanho = tamanhoDoSobrenome () ; 


A(String nome) { 
this.nome = nome; 


int tamanhoDoSobrenome() 1 
return sobrenome.length(); 


AQ 4 
E; 


Cuidado ao invocar métodos no construtor e variáveis estarem nulas: 


class Teste { 


String nome; 

Teste() { 
testaTamanho(); // NullPointerException 
nome = "aprendendo"; 


private void testaTamanho() { 
System.out.println(nome.length()); 


public static void main(Stringl[] args) 1 
new Teste(); 


E mais cuidado ainda caso isso ocorra por causa de sobrescrita de método, 





em que também poderemos ter essa Exception: 
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class Base { 
String nome; 


Base() { 
testa(); 
nome = "aprendendo"; 


void testa() { 
System.out.println("testa"); 


} 


class Teste extends Base { 
void testa() { 
System.out.println(nome.length()); 
F 
public static void main(String[] args) { 
new Teste(); 


Já seo método testa for privado, como o binding da chamada ao método 
é feito em compilação, o método invocado pelo construtor é o da classe mãe, 





sem dara Exception: 


class Base { 
String nome; 


Base() { 

testa(); 

nome = "aprendendo"; 
j 


private void testa() { 
System.out.println("testa"); 


} 


class Teste extends Base { 
void testa() { 
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System.out.println(nome.length()); 

J 

public static void main(String[] args) { 
new Teste(); 


Você pode entrar em loop infinito, cuidado, StackOverflow: 


class Teste { 
Teste() { 
new Teste(); 
} 
public static void main(String[] args) { 
new Teste(); 


Construtores podem ser de todos os tipos de modificadores de acesso: 
private, protected, default e public. 





E comum criar um construtor privado e um método estático para criar 
seu objeto: 


class Teste { 
private Teste() { 
} 


public static Teste cria() { 
return new Teste(); 


Tenha muito cuidado com um método com nome do construtor. Se colo- 
carum void na frente, vira um método: 


class Teste { 
void Teste() { 
System.out.printin("Construindo'"); 


} 
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public static void main(Stringl[] args) 1 
new Teste(); 
// não imprime nada, definimos um método e não o construtor 


new Teste().Teste(); 
// agora imprime Construindo 


Existem também blocos de inicialização que não são cobrados na prova. 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 final String n; 

3 AO 1 

4 a(); 

5 n = "aprendendo"; 

6 } 

: void a() { 

8 System.out.printin("testa!); 

9 } 

1 + 

u class B extends A { 

12 void a() { 

3 System.out.printin(n.length()); 
14 } 

15 public static void main(String[] args) { 
16 new B(); 

17 F} 

18 + 


a) Não compila. 

b) Compila e imprime “testa”, 

c) Compila e imprime length. 
d) Compila e dá exception. 


e) Compila e não imprime nada. 
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8.5 CRIE E SOBRECARREGUE CONSTRUTORES 
Construtores também podem ser sobrecarregados: 


class Teste { 
public Teste() + 
} 
public Teste(int i) { 
} 


Cuidado com os exemplos de sobrecarga com varargs, como vimos 
antes, e no caso de herança. 

Quando existem dois construtores na mesma classe, um construtor pode 
chamar o outro através da chamada this. Note que loops não compilam: 


class Teste { 
public Teste() 1 
System. out .println("construtor simples"); 


} 

public Teste(int i) { 
this(); 

} 


public Teste(String s) { 
this(s, s); // não compila, loop 
E; 
public Teste(String s, String s2) 1 
this(s); // não compila, loop 


Temos que tomar cuidado com sobrecarga da mesma maneira que 
tomamos cuidado com sobrecarga de métodos: os construtores invocados 
seguem as mesmas regras que as de métodos. 

Quando um método utiliza varargs, se ele possui uma variação do 
método sem nenhum argumento e invocarmos sem argumento, ele chamará 
o método sem argumentos (para manter compatibilidade com versões ante- 
riores do Java): 
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void desativa(Cliente... clientes) { 
System.out.println('varargs"); 

} 

void desativa() { 
System.out.println("sem argumento"); 


} 
void metodo() { 
desativa(); // imprime sem argumento 


A instrução this do construtor deve ser sempre a primeira dentro do 
construtor: 


class Teste { 
Teste() { 
String valor = "valor..."; 
this(valor); // não compila 


Teste(String s) { 
System.out.println(s); 


public static void main(String[] args) { 
new Teste(); 


Justo por isso não é possível ter duas chamadas a this: 


class Teste { 
Teste() { 
this(valor); 
this(valor); // não compila 


Teste(String s) { 
System.out.println(s); 
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public static void main(Stringl[] args) 1 
new Teste(); 


A instrução this pode envolver instruções: 


class Teste { 
Teste() { 
this(valor (0); 


private static String valor() { 
return "valor..."; 


Teste(String s) 1 
System.out.printin(s); 


public static void main(Stringl[] args) { 
new Teste(); 


A instrução não pode ser um método da própria classe, pois o objeto não 
foi construído ainda: 


class Teste { 
Teste() { 
this(valor()); // valor não é estático, não compila 


private String valor() { 
return '"valor..."; 


Teste(String s) 1 
System.out.printin(s); 
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public static void main(Stringl] args) { 
new Teste(); 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { B() { this(1); } B(int i) { this(); 55 
2 class A { 


3 public static void main(String[] args) { 
4 new B(); 

5 } 

6} 


a) Não compila. 
b) Compila e joga exception. 


c) Compila e não imprime nada. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B() { B(A a) {} BO 1) 5 
class C() { C(@B b) {1} CO 1) + 


n 


N 


3 class A { 

4 public static void main(String[] args) { 
5 new A(); new B(); new C(); 

6 } 

7 5 


a) Não compila ao invocar o construtor de A. 
b) Não compila ao invocar o construtor de B. 
c) Não compila ao invocar o construtor de c. 
d) Não compila na definição das classes Be c. 
e) Compila e joga exception. 


f) Compila e não imprime nada. 
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3) 


1 


4) 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B { B(A a) 1) BO 1) 5 
class C(C(Bb) Q} CO Q} 
class A { 
public static void main(Stringl] args) 1 
new A(); new B(); new C(); 


a) Não compila ao invocar o construtor de A. 
b) Não compila ao invocar o construtor de B. 
c) Não compila ao invocar o construtor de c. 
d) Não compila na definição das classes Be C. 
e) Compila e joga exception. 


f) Compila e não imprime nada. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B { B(A a) {} BO 1) 5 
class C { C(B b) 1) CO 15 
class A { 
public static void main(Stringl] args) 1 
new C(new B(new A())); 


a) Não compila ao invocar o construtor de A. 
b) Não compila ao invocar o construtor de B. 
c) Não compila ao invocar o construtor de c. 
d) Não compila na definição das classes Be c. 
e) Compila e joga exception. 


f) Compila e não imprime nada. 
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5) 


1 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B { B(A a) {new C(); ) BO { new C(this);) } 
class C { C(B b) {new B(new A());} CO {new BO;) + 
class A { 
public static void main(String[] args) { 
new C(new B(new A())); 


a) Não compila ao invocar o construtor de A. 
b) Não compila ao invocar o construtor de B. 
c) Não compila ao invocar o construtor de C. 
d) Não compila na definição das classes Be c. 
e) Compila e joga exception. 


f) Compila e não imprime nada. 


8.6 APLIQUE MODIFICADORES DE ACESSO 


Os modificadores de acesso, ou modificadores de visibilidade, servem para 


definir quais partes de cada classe (ou se uma classe inteira) estão visíveis 


para serem utilizadas por outras classes do sistema. Só é permitido usar um 


único modificador de acesso por vez: 


private public int x; // não compila 


O Java possui os seguintes modificadores de acesso: 
e public 

e protected 

e Nenhum modificador, chamado de default 


e private 
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Classes e interfaces só aceitam os modificadores publicou default. 

Membros (construtores, métodos e variáveis) podem receber qualquer 
um dos quatro modificadores. 

Variáveis locais (declaradas dentro do corpo de um método ou constru- 
tor) e parâmetros não podem receber nenhum modificador de acesso, mas 
podem receber outros modificadores. 





Top LEVEL CLASSES E INNER CLASSES 


Classes internas (nested classes ou inner classes) são classes que são 
declaradas dentro de outras classes. Esse tipo de classe pode receber qual- 
quer modificador de acesso, já que são consideradas membros da classe 
onde foram declaradas (top level class). 

Nesta certificação não são cobradas classes internas, apenas top level 
classes. 











Para entender como os modificadores funcionam, vamos imaginar as 


seguintes classes: 


package forma; 


class Forma 
double lado; 
double getArea(){ 
return O; 


} 


package forma; 


class Quadrado extends Forma{} 


package forma.outro; 
import forma.*; 


class Triangulo extends Formal) 
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Public 


O modificador public é o menos restritivo de todos. Classes, interfaces 
e membros marcados com esse modificador podem ser acessados de qual- 
quer componente, em qualquer pacote. Vamos alterar nossa classe Forma, 
marcando-a e todos seus membros com o modificador public: 


package forma; 


public class Formal 
public double lado; 
public double getArea(){ 
return O; 


Agora vamos fazer um teste: 


package forma.outro; 
import forma.*; 


public class TesteDutroPacote( 


public static void main(String... args) 
Forma f = new Forma(); //acesso a classe forma 
f.lado = 5.5; //acesso ao atributo lado 
f.getArea(); //acesso ao método gethrea() 


Repare que, mesmo nossa classe TesteOutroPacote estando em um 
pacote diferente da classe Forma, é possível acessar a classe e todos os mem- 
bros declarados como public. 


Protected 


Membros definidos com o modificador protected podem ser acessa- 
dos por classes e interfaces no mesmo pacote, e por qualquer classe que es- 
tenda aquela onde o membro foi definido, independente do pacote. 


Vamos modificar nossa classe Forma para entendermos melhor: 
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package forma; 


public class Forma{ 
protected double lado; // agora protected 
public double getArea(){} 


Com o modificador protected, nossa classe de testes em outro pacote 


não compila mais: 


package forma.outro; 
import forma.*; 


public class TesteDutroPacote( 


public static void main(String... args)( 
Forma f = new Forma(); 
f.lado = 5.5; // erro de compilação 
f.gethArea(); 


Se criarmos uma nova classe de teste no pacote forma, conseguimos 
acessar novamente o atributo: 


package forma; 
public class Teste{ 
public static void main(String... args){ 


Forma f = new Forma(); 
f.lado = 5.5; // compila normal, mesmo pacote 


Embora esteja em um pacote diferente, a classe Triangulo consegue 
acessar o atributo lado, já que ela estende da classe Forma: 


package forma.outro; 
import forma.*; 
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class Triangulo extends Formal 


public void imprimeLado 01 
//Como é uma classe filha, acessa 
//normalmente os membros protected da classe mãe. 


System.out.printin("0 Lado é " + lado); 


Agora repare que, se efetuarmos o casting do objeto atual para uma 
Forma, não podemos acessar seu lado: 


package outro; 
import forma.*; 


class Triangulo extends Formal 


public void imprimeLado 01 
// compila 


System.out.printin("0 Lado é " + lado); 


// não compila 
System.out.printin("0 Lado é " + ((Forma) this) .lado); 


Isso ocorre porque estamos dizendo que queremos acessar a variável 
membro lado de um objeto através de uma referência para este objeto, e 
não diretamente. Diretamente seria o uso puro do this ou nada. Nesse caso, 
após usar o this, usamos um casting, o que deixa o compilador perdido. 


Default 


Se não definirmos explicitamente qual o modificador de acesso, pode- 
mos dizer que aquele membro está usando o modificador default, também 
chamado de package private. Neste caso, os membros da classe só serão 
visíveis dentro do mesmo pacote: 
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package forma; 


public class Formal 
protected double lado; 
public double getArea(){ 


return O; 

} 

double getPerimetro(){ //default access 
return 0; 

} 


O método getPerimetro () só será visível para todas as classes do pa- 
cote forma. Nem mesmo a classe Triangulo que, apesar de herdar de 
Forma, está em outro pacote consegue ver o método. 


package outro; 
import forma.*; 


class Triangulo extends Forma{ 
public void imprimePerimetro(){ 


//Erro de compilação na linha abaixo 
System. out.println("0 Perímetro é " + getPerimetro()); 
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PALAVRA-CHAVE DEFAULT 


Lembre-se! A palavra-chave default é usada para definir a opção 
padrão em um bloco switch, ou para definir um valor inicial em uma 
Annotation. Usá-la em uma declaração de classe ou membro é inválido e 
causa um erro de compilação: 


| default class Bolaí //ERRO 
4 default String cor; // ERRO 


A partir do Java 8, a palavra default também pode ser usada para 
definir uma implementação inicial de um método. 











Mas e se declararmos uma classe com o modificador default? Isso 
vai fazer com que aquela classe só seja visível dentro do pacote onde foi 
declarada. Não importa quais modificadores os membros dessa classe ten- 
ham, se a própria classe não é visível fora de seu pacote, nenhum de seus 
membros é visível também. 


Veja a classe Quadrado, que está definida com o modificador default: 


package forma; 
class Quadrado extends Forma{} 


Veja o seguinte código, usando a classe TesteOutroPacote. Perceba 
que não é possível usar a classe Quadrado, mesmo importando todas as 
classes do pacote forma: 


package outro; 
import forma.*; 


public class TesteDutroPacote( 
public static void main(String... args) 
Quadrado q = new Quadrado(); // erro, esta classe não é 


//visível 
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LINHA COM ERRO DE COMPILAÇÃO 


Eventualmente, na prova, é perguntado em quais linhas ocorreram os 
erros de compilação. É bem importante prestar atenção nesse detalhe. 

Por exemplo, neste caso, o erro sempre acontecerá quando tentarmos 
acessar a classe Quadrado, que não é visível fora de seu pacote: 


package outro; 


//import de todas as classes PÚBLICAS do pacote, nenhum erro 
import forma.*; 


public class TesteDutroPacotef 


public static void main(String... args){ 
// erro na linha 8, Quadrado não é visível, pois não 
// & pública 
Quadrado q = new Quadrado (); 


O mesmo código pode apresentar erro em uma linha diferente, ape- 
nas mudando o import. Repare que o código a seguir dá erro nas duas 
linhas, tanto do import quanto na tentativa de uso: 


package outro; 
// erro na linha 3, não podemos importar classes não públicas 
import forma.Quadrado; 


public class TesteDutroPacote( 
public static void main(String... args) 


//Erro, pois Quadrado não é acessível. 
Quadrado q = new Quadrado (); 


É muito importante testar vários trechos de código, para ver exata- 
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Private 


private é o mais restritivo de todos os modificadores de acesso. Mem- 
bros definidos como private só podem ser acessados de dentro da classe e 
de nenhum outro lugar, independente de pacote ou herança: 


package forma; 


public class Formal 
protected double lado; 
public double getArea(){} 


//cor só pode ser acessada dentro da classe Forma, 
//nem as classe Quadrado e Triangulo conseguem acessar 
private String cor; 





PRIVATE E CLASSES ANINHADAS OU ANÔNIMAS 


Classes aninhadas ou anônimas podem acessar membros privados da 
classe onde estão contidas. Na certificação tais classes não são cobradas. 











Métodos privados e padrão não podem ser sobrescritos. Se uma classe 
o “sobrescreve”, ele simplesmente é um método novo, portanto não podemos 
dizer que é sobrescrita. Veremos isso mais a fundo na seção sobre sobrescrita. 
9.1 


Resumo das regras de visibilidade 


Todos os membros da classe com o modificador de private só podem 
ser acessados de dentro dela mesma. 

Todos os membros da classe sem nenhum modificador de visibilidade, 
ou seja, com visibilidade package-private, podem ser acessados de dentro da 
própria classe ou de dentro de qualquer outra classe, interface ou enum do 
mesmo pacote. 
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Todos os membros da classe com o modificador protected podem ser 
acessados: 


e de dentro da classe, ou de dentro de qualquer outra classe, interface ou 
enum do mesmo pacote; 


e de dentro de alguma classe que deriva direta ou indiretamente da classe, 
independente do pacote. O membro protected só pode ser chamado 
através da referência this, ou por uma referência que seja dessa classe 
filha. 


Todos os membros da classe com o modificador public podem ser aces- 
sados de qualquer lugar da aplicação. 

E não podemos ter classes/interfaces/enums top-level como private ou 
protected 

Uma classe é dita top-level se ela não foi definida dentro de outra classe, 
interface ou enum. Analogamente, são definidas as interfaces top-level e os 
enums top-level. 


1) Escolha a opção adequada ao tentar compilar e rodar o Teste. Arquivo 
no diretório atual: 


ı import modelo.Cliente; 
2 class Teste { 


3 public static void main(Stringl] args) 1 
4 new Cliente('"guilherme") . imprime(); 
5 } 

6} 


Arquivo no diretório modelo: 


n 


package modelo; 


public class Cliente { 


w 


4 private String nome; 

5 Cliente(String nome) { 
6 this.nome = nome; 
7 E; 
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8 public void imprime() { 

9 System.out.println (nome) ; 
10 } 

11 } 


a) Não compila: erro na classe Teste. 
b) Não compila: erro na classe Cliente. 
c) Erro de execução: método main. 


d) Roda e imprime “Guilherme”. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 private static int a(int b) { 

3 return b(b)-1; 

4 } 

5 private static int b(int b) É 

6 return b-1; 

7 } 

8 public static void main(Stringl] args) 1 
9 System.out.printin(new A() .a(5)); 
10 } 

11 } 


a) Não compila nas invocações de métodos. 
b) Não compila na declaração de variáveis e métodos. 
c) Compila e imprime 3. 


d) Compila e dá erro. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 private public int a(int b) { 
3 return b(b)-1; 

4 F} 

5 private static int b(int b) É 
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6 return b-1; 

7 } 

8 public static void main(Stringl[] args) 1 
9 System.out.printin(new A() .a(5)); 

10 } 

11 } 


a) Não compila nas invocações de métodos. 
b) Não compila na declaração de variáveis e métodos. 
c) Compila e imprime 3. 


d) Compila e dá erro. 


4) Escolha a opção adequada ao tentar compilar e rodar os arquivos a seguir, 
cada um em seu diretório adequado: 


package a; 


m 


N 


import b.*; 
3 public class A extends B { protected int a(String s) 
á {return 2;) + 


1 package b; 
2 import a.x; 


3 public class B { public int a(Object s) {return 1;J } 


import a.*; 


nm 


N 


import b.*; 


3 class A { 

4 public static void main(String[] args) { 
5 System.out.printin(new A().a("a")); 
6 } 

zJ 


a) Não compila. 
b) Imprime 1. 
c) Imprime 2. 


d) Erro em execução. 
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5) 


w 


w 


w 


6) 


N 


N 


w 


Escolha a opção adequada ao tentar compilar e rodar os arquivos a seguir, 
cada um em seu diretório adequado: 


package a; 

import b.*; 

public class A extends B { protected int a(String s) 
{return 2;} } 


package b; 
import a.*; 
public class B { public int a(Object s) {return 1;} } 


import a.*; 
import b.*; 
class C { 
public static void main(String[] args) { 
System.out.println(new A().a("a")); 


a) Não compila. 
b) Imprime 1. 
c) Imprime 2. 


d) Erro em execução. 


Escolha a opção adequada ao tentar compilar e rodar os arquivos a seguir, 
cada um em seu diretório adequado: 


package a; 

import b.x*; 

public class A extends B { protected int a(String s) 
{return 2;} } 


package b; 


import a.*; 
public class B { default int a(Object s) {return 1;} } 
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ı import a.*; 
2 import b.*; 


3 class C { 

4 public static void main(String[] args) { 
5 System.out.printin(new A().a("a")); 
6 } 

7} 


a) Não compila. 
b) Imprime. 
c) Imprime 2. 


d) Erro em execução. 


7) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B{ 

2 static int bs=0; 

3 final int b = ++bs; 

4 private BO {} 

5 static B b() { return new B(); + 
6 + 

7 class A { 

8 public static void main(Stringl] args) 1 
9 System.out.printin(B.b() .b); 
10 } 

nu } 


a) Não compila. 
b) Compila e imprime 1. 
c) Compila e imprime o. 


d) Compila e dá erro de execução. 
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8.7 APLIQUE PRINCÍPIOS DE ENCAPSULAMENTO A 
UMA CLASSE 


A assinatura de um método é o que realmente deve importar para o usuário 
de alguma classe. Segundo os bons princípios do encapsulamento, a imple- 
mentação dos métodos deve estar encapsulada e não deve fazer diferença para 
o usuário. 

O que é importante em uma classe é o que ela faz e não como ela faz. O 
que ela faz é definido pelos comportamentos expostos, ou seja, pelos métodos 
e suas assinaturas. 

O conjunto de assinaturas de métodos visíveis de uma classe é chamado de 
interface de uso. É através dessas operações que os usuários vão se comunicar 
com os objetos dessa classe. 

Mantendo os detalhes de implementação de nossas classes “escondidos”, 
evitamos que mudanças na forma de implementar uma lógica quebre vários 
pontos de nossa aplicação. 

Uma das formas mais simples de começar a encapsular o comportamento 
de uma classe é escondendo seus atributos. Podemos fazer isso facilmente 
usando a palavra-chave private: 


public class Pessoaí 
private String nome; 


Caso precisemos acessar um desses atributos a partir de outra classes, 
teremos que criar um método para liberar o acesso de leitura desse atrib- 
uto. Seguindo a especificação dos javabeans, esse método seria um getter. 
Da mesma forma, se precisarmos liberar a escrita de algum atributo, criamos 


um método setter : 


public class Pessoaí 
private String nome; 


public String getNome() + 
return nome; 
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public void setNome (String nome) 1 
this.nome = nome; 


Com essa abordagem, poderíamos fazer uma validação em nossos méto- 
dos, para evitar que nossos atributos fiquem com estado inválido. Por exem- 
plo, podemos verificar se o nome possui pelo menos 3 caracteres: 


public class Pessoaí 
private String nome; 
private String sobrenome; 


public String getNome() 1 
return nome; 


} 
public void setNome(String nome) { 
if (nome!= null && nome.trim().length() >= 3) 
this.nome = nome; 
else{ 
throw new IllegalArgumentException( 
"Nome deve possuir " + "pelo menos 3 caracteres"); 


Encapsulamento é muito mais do que atributos privados e getters e setters . 
Não é nosso foco aqui discutir boas práticas de programação, e sim o conhec- 
imento necessário para passar na prova. Em questões sobre encapsulamento 
sempre, fique atento à alternativa que esconde mais detalhes de implemen- 
tação da classe analisada. A prova pode utilizar tanto o termo encapsulation 
como information hiding para falar sobre encapsulamento (ou esconder in- 
formações). 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B{ 
2 private int b; 
3 public int getB() { return b; } 
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4 public void setB(int b) ( this.b= b; + 

5 5 

é class A { 

7 public static void main(String[] args) { 
8 new B() .setB(5); 

9 System.out.printin(new B() .getB(O)); 
10 } 

11 } 


a) Não compila. 
b) Compila e imprime o. 


c) Compila e imprime 5. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B{ 

2 private int b; 

3 public int getB() { return b; } 

4 public void setB(int b) { this.b= b; } 
5 } 

é class A { 

7 public static void main(String[] args) { 
8 B b = new B(); 

9 b.setB(5); 

10 System.out.printin(b.getB()); 

11 } 

n2 + 


a) Não compila. 
b) Compila e imprime o. 


c) Compila e imprime 5. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B{ 
2 private int b; 
3 public int getB() { return b; } 


251 


8.7. Aplique princípios de encapsulamento a uma classe Casa do Código 





4 public void setB(int b) { b= b; 5 

s + 

é class A { 

7 public static void main(String[] args) { 
8 B b = new B(); 

9 b.setB(5); 

10 System.out.printin(b.getB(O); 

nu } 

2 + 


a) Não compila. 
b) Compila e imprime o. 


c) Compila e imprime 5. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class BÍ 

2 int b; 

3 public void setB(int b) { b= b; 5 
43 

s class A { 

6 public static void main(Stringl] args) 1 
7 B b = new B(); 

8 b.setB(5); 

9 System.out.printin(b.b); 

10 } 

un + 


a) Não compila, pois não é possível ter setter sem getter. 
b) Compila e imprime o. 


c) Compila e imprime 5. 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class BÍ 
2 private final int b; 
3 B(int b) 1 this.b = b;} 
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4 public int getB() ( return b; 5 

5 public void setB(int b) { b= b; 5 

6} 

7 class A { 

8 public static void main(Stringl] args) 1 
P B b = new B(10); 

10 b.setB(5); 

u System.out.println(b.getB()); 

12 E; 

5 5 


a) Não compila. 
b) Compila e imprime o. 
c) Compila e imprime 5. 


d) Compila e imprime 10. 


8.8 DETERMINE O EFEITO QUE OCORRE COM REFER- 
ÊNCIAS A OBJETOS E A TIPOS PRIMITIVOS QUANDO 
SÃO PASSADOS A OUTROS MÉTODOS E SEUS VAL- 
ORES MUDAM 


As informações que queremos enviar para um método devem ser passadas 
como parâmetro. O dominio de como funciona a passagem de parâmetro é 
fundamental para a prova de certificação. 

O requisito para entender passagem de parâmetro no Java é saber como 
funciona a pilha de execução e o heap de objetos. 

A pilha de execução é o “lugar” onde são empilhados os métodos invoca- 
dos na mesma ordem em que foram chamados. 

O heap é o “lugar” onde são guardados os objetos criados durante a exe- 
cução. 


Considere o exemplo a seguir: 


class Teste { 
public static void main(String[] args) { 
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Casa do Código 
int i = 2; 
teste(i); 

+ 

private static void teste(lint i) | 
for (int j= 0; j < ip j+) { 

new String("j =" + j); 

} 

} 

E 
i=2 
teste j=0..1 

main [i=2 N4 main [i=2 

















Figura 8.1: Pilha de execução 





Figura 8.2: Heap 


A passagem de parâmetros é feita por cópia de valores. Dessa forma, 
mudanças nos valores das variáveis definidas na lista de parâmetros de um 


método não afetam variáveis de outros métodos. 


Passagem de parâmetros primitivos 


Veja o seguinte código: 
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class Teste { 
public static void main(Stringl[] args) 1 
int i = 2; 
teste(i); 
System.out.printin(i); 


static void teste(int i) É 
i = 3; 


Ao executar a classe Teste, será impresso o valor 2. É necessário perce- 
ber que as duas variáveis com o nome i estão em métodos diferentes. Há um 
i no main () eoutro i no teste (). Alterações em uma das variáveis não 
afetam o valor da outra. 


Passagem de parâmetros de referência 
Agora veja esta classe: 


class Teste { 
public static void main(String[] args) { 
Prova prova = new Prova(); 
prova.tempo = 100; 
teste (prova); 
System.out.println(prova.tempo); 


static void teste(Prova prova) { 
prova.tempo = 210; 


class Prova { 
double tempo; 


Esse exemplo é bem interessante e causa muita confusão. O que será im- 
presso na saída, ao executar a classe Teste, é o valor 210. Os dois métodos 
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têm variáveis com o mesmo nome ( prova). Essas variáveis são realmente 
independentes, ou seja, mudar o valor de uma não afeta o valor da outra. 
Por outro lado, como são variáveis não primitivas, elas guardam referên- 
cias e, neste caso, são referências que apontam para o mesmo objeto. Modifi- 
cações nesse objeto podem ser executadas através de ambas as referências. 


Pilha de Execução Heap 





Prova 


m 
tempo = 100 


ll 
| 


teste prova — 


Il 
. 
| 


main prova 


Figura 8.3: Passagem de parâmetros não primitivos 


Mas se eu trocar a referência, só estou trocando nesta variável local, e 
não no objeto referenciado, como no exemplo do teste2, em que estamos 
trocando somente a referência local e não o outro: 


class Prova { 
int tempo; 
Í 
class TestaReferenciaFPrimitivo { 
public static void main(Stringl[] args) { 
Prova prova 


new Prova(); 
100; 


prova.tempo 
teste(prova) 


ue 


System.out.printin(prova.tempo); 
teste2(prova); 


System.out.printin(prova.tempo); 


int i = 2; 
i = teste(i); 
System.out.println(i); 
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1) 


1 
2 
3 
4 


5 


2) 


É; 

static void teste2(Prova prova) { 
prova = new Prova(); 
prova.tempo = 520; 


static void teste(Prova prova) { 


prova.tempo = 210; 


static int teste(int i) É 
i = 5; 
System.out.printin(i); 
return i; 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class A { 

public static void main(Stringl] args) 1 
int i = 150; 
i = ++s(i); 
System.out.println(i); 

} 

static int s(int i) { 
return ++i; 


a) Não compila. 
b) Compila e imprime 150. 
c) Compila e imprime 151. 


d) Compila e imprime 152. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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ı class A { 

2 public static void main(Stringl[] args) 1 
3 int[] i = 1150, 151}; 

4 i = s(i); 

5 System.out.printin(i[1]); 

6 } 

; static int[] s(int[] i) { 

8 int[] j = {i[0], i[1]}; 

9 illl++; 

10 return j; 

11 } 

2 } 


a) Não compila. 

b) Compila e imprime 150. 
c) Compila e imprime 151. 
d) Compila e imprime 152. 


e) Compila e imprime 153. 
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Trabalhando com herança 


9.1 IMPLEMENTANDO HERANÇA 


Em Java, podemos usar herança simples entre classes com o extends. A 
nomenclatura usada é de classe mãe (parent class) e classe filha (child class), 
ou superclasse e subclasse. 

Herança entre classes permite que um código seja reaproveitado, de 
maneira que a classe filha reutilize o código da parte mãe, preocupando- 
se principalmente em sua especialização. A filha especializa a classe mais 
genérica. Herança em Java pode ser entre classes, reaproveitando membros, 


ou herança de uma interface, com a qual reaproveitamos interfaces de méto- 


dos. 


class Mae { 


} 
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class Filha extends Mae ( 
E; 
class Neta extends Filha { 
+ 


Vale lembrar que toda classe que não define de quem está herdando herda 
de Object: 


class Explicito extends Object { 
} 
class Implicito { 
// extends Object por padrão 
} 
class FilhoDeImplicito extends Implicito{ 
// também herda de Object, indiretamente 


Mas não podemos herdar de duas classes: 


class Simples1 {} 

class Simples2 {} 

class Complexa extends Simples1, Simples2 { 
// não compila 


Para podermos herdar de uma classe, a classe mãe precisa ser visível pela 
classe filha e pelo menos um de seus construtores também: 


class Pai { 
Pai(int x) { 
} 
} 
class Filho1 extends Pai{ 
// não compila pois o construtor padrão chama super() 
// e o Pai não tem construtor vazio 


} 
class Filho2 extends Pai{ 
Filho2() { 
super (15); //compila 
} 
} 
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Além disso, a classe mãe não pode ser final: 


final class Pai { 
} 
class Filho extends Pai { 
// não compila, Pai é final 
E; 
class Mae { 
F 
final class Filha extends Mae { 
// uma classe final pode estender de alguém, compila 


Herança de métodos e atributos 


Todos os métodos e atributos de uma classe mãe são herdados (indepen- 
dente das visibilidades). 


class X { 
int X; 
public void y() { 
} 
} 
class Y { 
// tenho um x, e o método y 
} 


Dependendo da visibilidade e das classes envolvidas, a classe filha não 
consegue enxergar o membro herdado. No exemplo a seguir, herdamos o 
atributo mas não o enxergamos diretamente. 


class X { 
private int x; 


public void setX(int x) { 
this.Xx = X; 
public int getX() { 


return X; 
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class Y extends X { 
public void metodo O { 
this.x = 5; // não compila: "x has private access in X" 


this.setX(10); // compila e altero o x herdado mas 
// não visível 


Métodos estáticos e herança 


Não existe herança de métodos estáticos. Mas quando herdamos de uma 
classe com métodos estáticos, podemos chamar o método da classe mãe us- 
ando o nome da filha (embora não seja uma boa prática): 


class W 1 
static void metodo() { 
} 

} 


class Z extends W { 
} 


class Teste { 
public static void main(String[] args) { 
Z.metodo(); // melhor seria escrever W.metodo() 


} 
} 
class W { 
public static void metodo() { 
System.out.println("w"); 
} 
} 


class Z extends W { 
public static void metodo() { 
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// não existe super em contexto estático, não compila 
super .metodo () ; 


Por não existir herança, o modificador abstract não é aceito em méto- 
dos estáticos. 

Podemos até escrever na classe filha um método estático de mesmo nome, 
mas isso não é sobrescrita (alguns chamam de redefinição): 


class W { 
public static void metodo() { 
System. out .println("w"); 


} 
class Z extends W { 
public static void metodo() { 
System. out.println("z"); 


} 
public class Teste { 
public static void main(String[] args) { 
System.out.printin(W.metodo()); // w 
System.out.println(Z.metodo()); // z 


Na verdade você até segue as regras de sobrescrita de método (visibilidade 
e retorno), mas no polimorfismo ele não funciona como métodos normais. 
Ele simplesmente funciona com o tipo da variável em tempo de compilação 
e não o tipo do objeto em tempo de execução: 


public class Teste { 
public static void main(String[] args) { 
Ww= new WC); 
w.metodo(); // w 


Z z= nev ZC); 
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z.metodo(); // z 


W zPolimorfadoComoW = z; 
zPolimorfadoComoW.metodo () ; 

// este último imprime w, 

// pois o binding é feito em compilação: 

// zPolimorfadoComoW.metodo é uma referencia 
// em compilação para W 





CONSTRUTORES E HERANÇA 


Não existe herança de construtores. O que existe é a classe filha 
chamar o construtor da mãe. 








SOBRESCRITA DE ATRIBUTOS 


Não existe sobrescrita de atributos. Podemos, sim, ter um atributo na 
classe filha com mesmo nome da mãe, mas não chamamos de sobrescrita. 
Nesses casos, o objeto vai ter 2 atributos diferentes, um da mãe (acessível 


com super) e um na filha (acessível com this). 











Object 


Em Java, toda classe é obrigada a usar a herança. Quando não escreve- 
mos extends explicitamente, estamos herdando de java.lang.Object 
automaticamente. 

Isso quer dizer que todo objeto em Java é um Object e, portanto, herda 
todos os métodos da classe Object (por isso esses são muito importantes). 

Há vários métodos em Object, que veremos ao longo do curso, mas o 
mais simples talvez seja o toString, que podemos sobrescrever em nossas 
classes para devolver alguma String que represente o objeto: 
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class Carro { 
String cor; 


public String toStringO { 
return "Um carro de cor " + this.cor; 


Temos que lembrar que o toString é chamado automaticamente para 
nós quando usamos o objeto no contexto de String: 


Carro c = new Carro(); 
c.cor = "Verde"; 


System.out.printin(c); // chama toString 


String s = "Mensagem: " + c; // chama toString 
System.out.printin(s); 


1) class A { 
public void metodo(long 1) { 
F 
} 


class B extends A{ 
protected void metodo(int i) { 


} 


Compila? 


2) import java.iox; 
class Veiculo 1 
protected void liga () throws IOException (J 
} 
class Carro extends Veiculo { 
public void liga() throws FileNotFoundException {} 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


265 


. Implementando herança Casa do Código 





n 


N 


w 


4) 


m 


N 


w 


5) 


ï 
2 


3 


class B extends C { int m(int a) { return 1; }} 
class C extends A { int m(double b) { return 3; } + 
class A extends B { 
int m(String c) { return 3; } 
public static void main(String[] args) { 
System.out.printin(new CO .m(3)); 


a) O código não compila. 
b) Compila e imprime. 
c) Compila e imprime 2. 
d) Compila e imprime 3. 


e) Compila e imprime 1, 2, 3. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B { int m(int a) { return 1; ) + 
class C { int m(double b) { return 2; } } 
class A extends B, C{ 
public static void main(String[] args) { 
System.out.println(new C().m(3)); 


a) O código não compila. 
b) Compila e imprime 1. 
c) Compila e imprime 2. 


d) Compila e imprime ı, 2. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B { private B() {} static B B(String s) 
{ return new BO; }} 
class A { 
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4 public static void main(Stringl] args) 1 
5 B b = B.B("t"); 

6 + 

75 


a) Não compila. 
b) Compila e imprime “t”. 
c) Compila e não imprime nada. 


d) Compila e joga uma exception. 
6) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
ı class B { private B() {} static B B(String s) 


5 { return new BO; ) 5) 
3 class A extends B { 


4 public static void main(String[] args) { 
5 B b = B.B("t"); 

6 } 

75 


a) Não compila. 
b) Compila e imprime “t”. 
c) Compila e não imprime nada. 


d) Compila e joga uma exception. 


7) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { 

2 private String s; 

3 protected BO) 1) 

4 static A B(String s) { 
š return new A(); 

6 F 

75 

s class A extends B { 

9 A(String s) 1 
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10 this.s = sS; 

11 } 

12 public static void main(String[] args) { 
= Bb=A.B("t'!); 

14 System.out.printin(b.s); 

15 } 

16 + 


a) Não compila. 
b) Compila e imprime “t. 
c) Compila e não imprime nada. 


d) Compila e joga uma exception. 


8) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { 

2 protected String s; 

3 protected BO) {} 

4 static A B(String s) { 

5 return new A(); 

6 } 

7} 

s class A extends B { 

9 A(String s) { 

10 this.s = sS; 

nu } 

12 public static void main(String[] args) { 
= b= ABU 

14 System.out.printin(b.s); 
15 } 

16 + 


a) Não compila. 
b) Compila e imprime “t. 
c) Compila e não imprime nada. 


d) Compila e joga uma exception. 
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9.2 DESENVOLVA CÓDIGO QUE MOSTRA O USO DE 
POLIMORFISMO 


Reescrita ou sobrescrita é a maneira como uma subclasse pode redefinir o 
comportamento de um método que foi herdado de uma das suas superclasses 
(direta ou indiretamente). 


class Veiculo 1 
public void liga() 1 
System.out.println("Veiculo está sendo ligado!"); 


class Carro extends Veiculo { 
public void liga() { 
System.out.println("Carro está sendo ligado!"); 


Agora considere: 


public class Teste{ 
public static void main(String [] args)( 
Veiculo v = new Carro(); 
v.ligaO; 


O método chamado aqui será o da classe Carro, independente de a refer- 
ência ser do tipo Veiculo (o que importa é o objeto). 

Qual método será executado é descoberto em tempo de execução (a assi- 
natura é decidida em tempo de compilação!), isso é a chamada virtual de 
método (virtual method invocation). 


Para reescrever um método, é necessário: 


e exatamente o mesmo nome; 


e os parâmetros têm que ser iguais em tipo e ordem (nomes podem mu- 


dar); 
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e retorno do método deve ser igual ou mais específico que o da mãe; 
e visibilidade deve ser igual ou maior que o da mãe; 
e exceptions lançadas devem ser iguais ou menos que na mãe; 


e método na mãe não pode ser final. 


Se essas regras não forem respeitadas, pode haver um erro de compilação, 
ou o método declarado não será considerado uma reescrita do método. 

A regra sobre visibilidade é: um método reescrito só pode ter visibilidade 
maior ou igual à do método que está sendo reescrito. (Essa não é uma regra 
mágica! Faz todo o sentido; pense um pouco sobre o que poderia acontecer 
se essa regra não existisse). 

O código a seguir não compila, pois ligar é público na classe mãe, então 
só pode ser reescrito com visibilidade pública: 


class Veiculo 1 
public void liga() { 
System.out.println("Veiculo esta sendo ligado!"); 


class Carro extends Veiculo 1 
protected void liga() { 
System.out.println("Carro esta sendo ligado!"); 


Muito cuidado com interfaces, pois a definição de um método é, por 
padrão, public e o exercício pode apresentar uma pegadinha de compi- 
lação: 


interface A { 


void a(); 

E; 

class B implements A { 
void a() { 


// não compila, o método deveria ser público 
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} 
class C implements A { 
public void a() { 
// compila 


Estranhamente, um método sobrescrito pode ser abstrato, dizendo para 
o compilador que quem herdar dessa classe terá que sobrescrever o método 
original: 


class A { 
void a() 1 
} 
} 
abstract class B extends A { 
abstract void a(); // sobrescrevendo como abstrato 


class C extends B{ 
// não compila, não redefiniu a 


class D extends B{ 
void a() 1 
// compila pois redefiniu a 


Sobre o retorno covariante: permite que a classe filha tenha um retorno 
igual ou mais específico polimorficamente (um subtipo). 

Cuidado! O retorno covariante não vale para tipos primitivos. Um exem- 
plo de retorno covariante: 


class A { 
List<String> metodo () 1 
// devolve lista 
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class B extends A { 
ArrayList<String> metodo() { 
// devolve array list 


Outra regra importante sobre reescrita é a assinatura em relação ao lança- 
mento de exceções ( throws). Um método reescrito só pode lançar as mesmas 
exceções checked ou menos que o métodos que está sendo reescrito (quanto 
às unchecked, não há regras e sempre podemos lançar quantas quisermos). 


import java.sql.SQLException; 
import java.io.I0Exception; 


class AÍ 
public void metodo () throws SQLException, IOException { 
E; 


class B extends A { 
public void metodo () throws IOException { 
} 


Esse código compila, pois o método na classe A lança menos exceções 
que na classe mãe, respeitando a regra. Já o código a seguir não compila: 


import java.sql.SQLException; 
import java.io.I0Exception; 


class A { 
public void metodo () throws SQLException { 
} 

E; 


class B extends A { 
public void metodo () throws IOException { 
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Apesar de ambos os métodos lançarem apenas uma exceção, não é isso 
que importa, pois elas são diferentes. Outro caso que não compila: 


import java.io.I0Exception; 


class A { 
public void metodo () throws I0Exception 1 
} 

} 


class B extends A { 
public void metodo () throws Exception { 


} 








Exception é muito mais que IOException. 

Repare que, quando dizemos menos exceções que na mãe, isso indica não 
apenas quantidade, mas também devemos considerar o polimorfismo. Se tro- 
carmos o exemplo anterior, compilamos: 


import java.io.Il0Exception; 


class A { 
public void metodo () throws Exception { 
} 

J 


class B extends A { 


public void metodo () throws IOException { 
} 





Compila, pois I0Exception é mais específico que Exception na ár- 





vore de herança. 
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Polimorfismo e chamadas de métodos 
Imagine as classes: 


class Veiculo 1 
void liga() { 
System.out.printin("ligando o veiculo"); 


} 


class Carro extends Veiculo { 
void liga() { 
System.out.println("ligando o carro"); 


void desliga() { 
} 


Se tivermos um objeto do tipo Carro com uma referência do tipo 
Carro, ou seja, sem usar polimorfismo, podemos fazer: 


Carro c = new Carro(); 
c.liga(); // ligando o carro 
c.desliga(); 


Conseguimos chamar os dois métodos. E, como estamos trabalhando 
com sobrescrita, o método liga chamado é o da classe filha Carro. 


Mas e se usarmos polimorfismo e a referência para Veiculo? 


Veiculo v = new Carro(); 
v.liga(); // ligando o carro? 
v.desliga(); 


Vamos linha por linha: primeiro, podemos chamar um Carro de 
Veiculo porque ele é um (compila sem problemas). Podemos também 
chamar o método liga pois ambas as classes o possuem. Mas o método 
que será invocado será o da classe filha, o sobrescrito. 

Já a chamada ao método desliga não compilará, porque ele não está 
definido na classe veiculo. Como a referência é desse tipo, o método (que 
existe no objeto) não é visível. 
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A regra é: para saber se um método de um objeto pode ser chamado, 
olhamos para o tipo da referência em tempo de compilação. Para realmente 
chamar o método em tempo de execução, devemos olhar para o objeto ao que 
demos new. 

Essa regra faz sentido quando pensamos em um método polimórfico 
como o seguinte: 


void metodo (Veiculo v) { 
v.liga(); // compila 
v.desliga(); // não compila 


Se passarmos um objeto Carro para o método, teoricamente ambas 
as chamadas funcionariam, já que a classe possui tanto o liga quanto o 
desliga. 

Mas imagine uma classe Moto que não tem o método desliga. Como 
Moto éum Veiculo, podemos passar como argumento. O que aconteceria 
se pudéssemos ter chamado o desliga na referência Veiculo? Alguma 
coisa estaria errada. 

Portanto, a regra geral é que somente podemos acessar os métodos de 
acordo com o tipo da referência, pois a verificação da existência do método 
é feita em compilação. Mas qual o método que será invocado, isso será con- 
ferido dinamicamente, em execução. 

Um ponto muito importante é que o compilador nunca sabe o valor das 
variáveis depois da linha que as cria. Ou seja, o compilador não sabe se esta- 
mos passando um Carro ou uma Moto. O que ele sabe é apenas o tipo da 
variável; no caso, Veiculo. Ecomo Veiculo não tem o método desliga, 
o código não pode compilar. 


this, super e sobrescrita de métodos 


Na ocasião em que um método foi sobrescrito, podemos utilizar as 
palavras-chave super e this para deixar explícito qual método desejamos 
invocar: 


class A { 
public void metodo() 1 
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System.out.println("a"); 


} 
class B extends A { 
public void metodo() { 
System.out.println("b"); 
super.metodo(); // imprime a 


public void metodo2() { 
metodo(); // imprime b, a 
super.metodo(); // imprime a 


E se eu invocar o segundo método na primeira classe? Semo this? 


class M 
public void metodo() { 
System.out.println("a"); 
metodo2(); 
j 
public void metodo2() { 
System.out.println("metodo 2 do pai"); 


} 
class B extends A { 
public void metodo() { 
System. out.println("b"); 
super .metodo () ; 
} 
public void metodo2() { 
System. out.println("c"); 
metodo(); 
super .metodo () ; 
} 
public static void main(String[] args) { 
new B() .metodo2(); 
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O Java entra em loop infinito, uma vez que o método será invocado no 
objeto. Então faremos o lookup do metodo2 dinamicamente, encontrando 
o metodo2 que chama metodo, que chama metodo do pai, que chama no- 
vamente metodo2. Note que o lookup dos métodos, o binding dos métodos, 
é feito em execução mesmo se invocarmos dentro de um próprio objeto. Até 
mesmo o uso da palavra-chave this não evitaria isso, causando o loop: 


class AM 
public void metodo() 1 
System.out.printlin("a"); 
this.metodo2(); 
} 
public void metodo2() { 
System.out.println('metodo 2 do pai"); 


} 
class B extends A { 
public void metodo() { 
System. out.println("b"); 
super .metodo () ; 
} 
public void metodo2() { 
System. out.println("c"); 
metodo(); 
super .metodo () ; 
public static void main(Stringl[] args) { 
new B() .metodo2(); 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { 

2 void x() throws IOException { 
3 System.out.println("c"); 
4 } 

5 } 
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6 


2) 


1 
2 
3 
4 
5 
6 


7 


8 


class C extends B { 
void x() throws FileNotFoundException { 
System.out.printin("b''); 


} 
E; 
class A { 
public static void main(String[] args) { 
new C().x(); 
} 
} 


a) Não compila. 

b) Compila e imprime “b. 

c) Compila e imprime ‘c$ 

d) Compila e não imprime nada. 
e) Compila e dá exception. 


f) Compila e entra em loop. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


import java.io.%*; 
class B { 
void x() throws IOException { 
System. out.println("c"); 


} 
class C extends B { 
void x() throws FileNotFoundException { 
System. out.println("b"); 


E; 
} 
class A { 
public static void main(String[] args) throws IOException { 
new C().x(); 
} 
E; 
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a) Não compila. 

b) Compila e imprime ‘bi. 

c) Compila e imprime 'cí 

d) Compila e não imprime nada. 
e) Compila e dá exception. 


f) Compila e entra em loop. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı import java.io.*; 


2 class B { 

3 void x(double i) throws IDException { 

4 System.out.println("c"); 

5 } 

6} 

7 class C extends B { 

8 void x(int i) throws FileNotFoundException { 
9 System.out.printlin("b'); 

10 } 

11 } 

» class AÍ 

3 public static void main(String[] args) throws IOException { 
14 new C().x(3.2); 

15 } 

16 } 


a) Não compila. 

b) Compila e imprime ‘b. 

c) Compila e imprime ‘ci. 

d) Compila e não imprime nada. 
e) Compila e dá exception. 


f) Compila e entra em loop. 
4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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nm 


N 


w 


import java.io.*; 
class B { 
void x(double i) throws IOException { 
System. out.println("c"); 


} 
} 
class C { 
void x(int i) throws FileNotFoundException { 
System.out.printin("b''); 
} 
} 
class A { 
public static void main(String[] args) throws IOException { 
new C().x(3.2); 
} 
} 


a) Não compila. 

b) Compila e imprime “bí 

c) Compila e imprime ‘c$. 

d) Compila e não imprime nada. 
e) Compila e dá exception. 


f) Compila e entra em loop. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


import java.io.%*; 
interface B { 
public void x(double i) throws IOException { 
System. out.println("c"); 


} 


class C implements B { 
public void x(int i) throws FileNotFoundException { 


System. out.println("b"); 
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» class A { 

3 public static void main(String[] args) throws IOException { 
14 new CO .x(3); 

15 } 

16 } 


a) Não compila. 

b) Compila e imprime ‘b. 

c) Compila e imprime ‘ci. 

d) Compila e não imprime nada. 
e) Compila e dá exception. 


f) Compila e entra em loop. 
6) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
import java.io.*; 


class B { 
void x(int i) throws IOException { 


N 


w 


4 System.out.println("c"); 

5 } 

6} 

7 abstract class C extends B throws IOException { 
8 abstract void x(int i); 

9 5 


1 abstract class D extends C { 
u void x(int i) throws IOException { 


12 System.out.println("d"); 

13 F 

14 } 

5 class E extends D { 

16 + 

y class AÍ 

18 public static void main(Stringl[] args) throws IOException { 
19 new E() .x(32); 

20 } 

21 } 
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a) Não compila. 

b) Compila e imprime 'b. 

c) Compila e imprime ‘c$ 

d) Compila e imprime ‘di. 

e) Compila e não imprime nada. 
f) Compila e dá exception. 


g) Compila e entra em loop. 


7) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı import java.io.*; 


2 class B { 

3 void x(int i) throws IOException { 
4 if(i<0) return; 

5 x(-1); 

6 System.out.printin('c''); 

7 } 

8 + 

o abstract class C extends B 1 

10 void x(int i) throws IOException { 
u System.out.printin("b''); 

12 super.x(i); 

3 } 

14 > 

s abstract class D extends C { 

16 void x(int i) throws IOException { 
i7 super.x (i); 

18 } 

19 } 

2 class E extends D { 

21 } 

22 class A { 

23 public static void main(String[] args) throws IOException { 
2 new E().x(32); 

25 } 

2 } 
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a) Não compila. 

b) Compila e imprime ‘bi. 

c) Compila e imprime 'cí 

d) Compila e imprime 'd'. 

e) Compila e não imprime nada. 
f) Compila e dá exception. 


g) Compila e entra em loop. 


8) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı import java.io.x*; 


2 class B { 

3 void x(int i) throws IOException { 
4 if(i<0) return; 

5 this.x(-1); 

6 System.out.printlin(''c''); 

7 } 

8 + 

ə abstract class C extends B { 

10 void x(int i) throws IOException { 
u System.out.println("b"); 

12 super.x(i); 

13 F 

14 > 

5 abstract class D extends C { 

16 void x(int i) throws IOException { 
17 super.x(i); 

18 } 

19 } 

2 class E extends D { 

21 } 

» class A { 

23 public static void main(String[] args) throws IOException { 
34 new E() .x(32); 

25 } 

26 } 
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a) Não compila. 

b) Compila e imprime 'b. 

c) Compila e imprime ‘c$. 

d) Compila e imprime “b;'cí 

e) Compila e não imprime nada. 
f) Compila e dá exception. 


g) Compila e entra em loop. 


9) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı import java.io.x*; 


2 class B { 

3 void x(int i) throws IOException { 
4 if (i<0) return; 

5 super.x(-1); 

6 System.out.printin('c''); 

7 } 

s } 

ə abstract class C extends B { 

10 void x(int i) throws IOException { 
u System.out.printin("b''); 

12 super.x(i); 

3 } 

14 > 

5 abstract class D extends C { 

16 void x(int i) throws IOException { 
i7 super.x(i); 

18 } 

19 } 

2 class E extends D { 

21 } 

2 class A { 

23 public static void main(String[] args) throws IOException { 
2 new E().x(32); 

25 } 

2 } 
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a) Não compila. 

b) Compila e imprime ‘b. 

c) Compila e imprime ‘că. 

d) Compila e imprime “b;'c. 

e) Compila e não imprime nada. 
f) Compila e dá exception. 


g) Compila e entra em loop. 


9.3 DIFERENCIE ENTRE O TIPO DE UMA REFERÊNCIA E 
O TIPO DE UM OBJETO 


Sempre que estendemos alguma classe ou implementamos alguma interface, 
estamos relacionando nossa classe com a classe mãe ou interface usando um 
relacionamento chamado de é um. 

Se Carro extends Veiculo, dizemos que Carro é um Veiculo. Ou se 
ArrayList implements List dizemos que ArrayList é um List. 

O relacionamento de é um é um dos recursos mais poderosos da orien- 
tação a objetos. E é chamado formalmente de polimorfismo. 

Polimorfismo é a capacidade que temos de referenciar um objeto de for- 
mas diferentes, segundo seus relacionamentos de é um. 


Em especial, usamos polimorfismo quando escrevemos: 


Veiculo v = new Carro(); 
List 1 = new ArrayList(); 


As heranças e implementações vão formando uma árvore que terá sempre 
como raiz a classe Object. Assim, direta ou indiretamente, todo objeto é um 
Object. 

O polimorfismo pode ser aplicado à passagem de parâmetros (e é aí que 
está seu grande poder). Imagine as classes: 


class Veiculo {} 
class Carro extends Veiculo {} 
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class Moto extends Veiculo (> 
class Onibus extends Veiculo {} 
class Conversivel extends Carro {} 


Se temos um método que recebe Veiculo, podemos passar qualquer um 
daqueles objetos: 


void metodo (Veiculo v) É 


metodo (new Carro()); 
metodo (new Moto()); 

metodo (new Onibus()); 
metodo (new Veiculo()); 
metodo (new Conversivel()); 


Dessa forma, conseguimos obter um forte reaproveitamento de código. 

Repare que, quando usamos polimorfismo, estamos mudando o tipo da 
referência, mas nunca o tipo do objeto. Em Java, objetos nunca mudam seu 
tipo, que é aquele onde demos new. O que fazemos é chamar (referenciar) o 
objeto de várias formas diferentes. Chamar de várias formas... é o polimor- 
fismo. 

Podemos referenciar um objeto pelo seu próprio tipo, por uma de suas 
classes pai, ou por qualquer interface implementada por ele, direta ou indire- 
tamente: 


interface A {} 
interface B {} 
class C implements A {} 
class D extends C implements B {} 
public class Teste { 
public static void main(String[] args) { 


// mesmo tipo, compila 
D d = new D(); 
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// D extends C, todo D é um C, compila 
C c= new D(); 
C c2 = d; 


// D implements B, todo D implementa B, compila 
B b= new D(); 
B b2 = d; 


// D implements A indiretamente, compila 
A a = new D0); 


A a2 = a; 

D d2 = new C(); // não, C não é D, não compila 

D d3 = new D(); 

C c3 = d3; // compila 

D d4 = c3; // não compila, por mais que o ser humano 


// saiba, em execução, nem todo C é um D. 


E como funciona o acesso às variáveis membro e aos métodos? Se temos 
uma referência para a classe mãe, não importa o que o valor seja em tempo 
de execução, o compilador não conhece o tempo de execução, então ele só 
compila chamadas aos métodos definidos na classe mãe: 


class Veiculo { 
public void liga() { } 
} 
class Carro { 
public void mudaMarcha() {} 


// teste 
Veiculo v = new Veiculo(); 


v.liga(); // compila 


Carro c = new Carro(); // ok 
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c.mudaMarcha(); // compila 


Veiculo v2 = c; 
v2.liga(); // todo veiculo tem método liga, compila 
v2.mudaMarcha(); // não compila, nem todo veiculo tem 


Mesmo em casos em que “achamos” que todo veículo tem, se o método 
não foi definido na classe de referência, o codigo não compila: 


abstract class Veiculo { 
public void liga() { } 
} 
class Carro { 
public void desliga() { } 
} 
class Moto { 
public void desliga() { } 


Carro c = new Carro(); // ok 
c.desliga(); // compila 


Veiculo v2 = c; 
v2.desliga(); // não compila, Veiculo não tem o método desliga 
// definido 


O mesmo valerá para variáveis membro: 


class Veiculo { 
int velocidade; 
} 
class Carro { 
int marcha; 


// teste 
Veiculo v = new Veiculo(); 
v.velocidade = 3; // compila 


Carro c = new Carro(); // ok 
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c.marcha = 1; // compila 


Veiculo v2 = c; 
v2.velocidade = 5; // compila 
v2.marcha = 7; // não compila 


E temos que cuidar de mais um caso específico: o que acontece se estamos 
trabalhando com pacotes distintos? 

Se o método da classe pai que está sendo sobrescrito é public, os méto- 
dos que sobrescrevem devem ser public, então não tem muita graça. 

Já se o método da classe pai é protected, os filhos são protected ou 
public, que também não tem graça, pois o filho mesmo em outro pacote já 
tinha acesso ao método do pai. 

Mas o que acontece se o método no pai é private e eu tento sobrescrevê- 
lo? Ou se o método é default e tento sobrescrevê-lo em outro pacote? 
O mesmo valerá tanto para private quando para modificador de escopo 
padrão: 


package financeiro; 
public class ContaFinanceira extends modelo.Conta { 
void fecha() 1 
System.out.println('fechando financeiro"); 


} 


package modelo; 
public class Conta { 
void fecha() { 
System.out.println('fechando conta normal"); 


Ao invocar o método fecha através de uma referência para Conta ou 
ContaFinanceira, o resultado será totalmente diferente: 


ContaFinanceira c = new ContaFinanceira(); 
c.fecha(); 

Conta d = c; 

d.fecha(); 
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O código não compila, dependendo do pacote onde ele está. Como as- 
sim? Acontece que o método não foi sobrescrito, a classe filha nem sabe da 
existência do método (privado ou default) do pai, portanto o que ela fez foi 
criar um método totalmente novo. 

Nesse caso, ao invocarmos o método durante compilação, o binding é feito 
para o método específico de cada uma delas, uma vez que são métodos to- 
talmente diferentes. Se o código está no pacote de modelo, a chamada ao 
método fecha de Conta compila e imprimiria normal. Se estivermos no 
pacote financeiro,achamadaao ContaFinanceira compila e imprime 
financeiro. 

Lembre-se que os métodos privados terão um efeito equivalente: eles só 
são vistos internamente à classe onde foram definidos. 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class D extends C { 


2 void x() { System.out.println(1); + 
3 5 

4 class C extends B 1 

5 void x() { System.out.println(2); + 
6 + 

7 class B { 

8 void x() { System.out.println(3); + 
5 void y(B b) 1 

10 b.xO; 

nu } 

12 void y(C b) 1 

5 exU 

14 F 

E void y(D b) 1 

16 d.x(); 

17 } 

18 + 

1 class A É 

20 public static void main(String[] args) { 
21 new B().y(new C()); 

22 } 

23 5 
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a) Não compila. 
b) Compila e imprime. 
c) Compila e imprime 2. 


d) Compila e imprime 3. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class D extends C { 


2 void x() { System.out.println(1); } 
3 5 

4 class C extends B 1 

5 void x() { System.out.println(2); + 
6 + 

7 class B { 

8 void x() { System.out.println(3); + 
ê void y(B b) 1 

10 b.xO; 

11 } 

12 void y(C c) { 

5 c.x(); 

14 } 

i void y(D d) { 

16 d.x(); 

17 } 

18 } 

1 class A { 

20 public static void main(String[] args) { 
21 new B().y(new C()); 

22 } 

23 } 


a) Não compila. 
b) Compila e imprime. 
c) Compila e imprime 2. 


d) Compila e imprime 3. 
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3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class D extends C { 


2 void x() { System.out.println(1); + 
35 

4 class C extends B { 

5 void x() { System.out.println(2); + 
6 + 

7 class B { 

8 void x() { System.out.println(3); + 
ò void y(B b) { 

10 b.xO; 

11 } 

2 + 

3 Class A { 

14 public static void main(Stringl] args) 1 
15 new B().y(new C()); 

16 } 

v } 


a) Não compila. 
b) Compila e imprime. 
c) Compila e imprime 2. 


d) Compila e imprime 3. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class D extends C { 


2 void x() { System.out.println(1); + 
3 void y(C b) { 

4 xO; 

5 } 

6 + 

7 class C extends B { 

8 void x() { System.out.println(2); + 
ə } 

1 class B { 

u void x() { System.out.println(3); + 
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12 void y(B b) { 

13 b.x(); 

14 + 

5 } 

6 class A { 

17 public static void main(Stringl] args) 1 
18 new B().y(new C()); 

19 } 

20 } 


a) Não compila. 
b) Compila e imprime. 
c) Compila e imprime 2. 


d) Compila e imprime 3. 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class D extends C { 


2 void x() { System.out.println(1); } 
3 void y(C b) { 

4 xO; 

5 } 

6 + 

7 class C extends B { 

8 void x() { System.out.println(2); + 
ə} 

1 class B { 

u void x() { System.out.println(3); + 
i2 void y(B b) { 

13 b.x(); 

14 } 

15 } 

6 class A { 

17 public static void main(String[] args) { 
18 new D().y(new C()); 

19 } 

20 } 
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6) 


n 


7) 


a) Não compila. 
b) Compila e imprime. 
c) Compila e imprime 2. 


d) Compila e imprime 3. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


package financeiro; 
public class ContaFinanceira extends modelo.Conta 1 
void fecha() { 
System.out.printin("fechando financeiro"); 


package modelo; 
public class Conta { 
void fecha() { 
System.out.printin("fechando conta normal"); 


package codigo; 
import modelo.x*; 
import financeiro.*; 
class A { 
public static void main(String[] args) { 
new Conta().fecha(); 


a) Não compila. 
b) Compila e roda jogando exception. 
c) Compila e roda, imprimindo ‘fechando financeiro. 


d) Compila e roda, imprimindo fechando conta normal. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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w 


8) 


package financeiro; 
public class ContaFinanceira extends modelo.Conta { 
void fecha() { 
System.out.println("fechando financeiro"); 


package modelo; 
public class Conta { 
public void fecha() { 
System.out.println("fechando conta normal"); 


package modelo; 
import modelo. *; 
import financeiro. *; 
class A { 
public static void main(String[] args) { 
new Conta().fecha(); 


a) Não compila. 
b) Compila e roda jogando exception. 
c) Compila e roda, imprimindo ‘fechando financeiro. 


d) Compila e roda, imprimindo ‘fechando conta normal‘. 

Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
package financeiro; 

public class ContaFinanceira extends modelo.Conta { 


void fecha() { 
System.out.println('fechando financeiro"); 


295 


9.3. Diferencie entre o tipo de uma referência e o tipo de um objeto Casa do Código 





package modelo; 
public class Conta 1 
protected void fecha() { 
System.out.printin("fechando conta normal"); 


ı package codigo; 
2 import modelo.x*; 
3 import financeiro.x*; 


4 class A { 

5 public static void main(Stringl] args) 1 
6 new Conta().fecha(); 

7 } 

8 + 


a) Não compila. 
b) Compila e roda jogando exception. 
c) Compila e roda, imprimindo fechando financeiro. 


d) Compila e roda, imprimindo fechando conta normal. 


9) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


package financeiro; 
public class ContaFinanceira extends modelo.Conta 1 
public void fecha() 1 
System.out.printin("fechando financeiro"); 


package modelo; 
public class Conta { 
public void fecha() 1 
System.out.printin('fechando conta normal"); 
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ı package codigo; 


2 class A { 

3 public static void main(String[] args) { 
4 new Conta().fecha(); 

5 } 

6 + 


a) Não compila. 
b) Compila e roda jogando exception. 
c) Compila e roda, imprimindo fechando financeiro. 


d) Compila e roda, imprimindo fechando conta normal‘. 


10) O que acontece com o código a seguir? 


interface Veiculo 1 
int getMarcha(); 
void liga(); 


abstract class Carro implements Veiculo { 
public void liga() { 
System.out.printin("ligado!"); 


class CarroConcreto extends Carro implements Veiculo 1 
public int getMarcha() { 
return 1; 


9.4 DETERMINE QUANDO É NECESSÁRIO FAZER CAST- 
ING 


As vezes, temos referências de um tipo mas sabemos que lá há um objeto de 
outro tipo, um mais específico: 
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public class Testel 
public static void main(String...args)( 
Object [] objetos = new Object [100]; 


String s = "certificacao"; 
objetos[0] = s; 


String recuperada = objetos[0]; 


O código acima não compila: 


Teste.java:3: incompatible types 
found : java. lang. Object 
required: java.lang.String 
String recuperada = objetos[0]; 


1 error 


Temos um array de referências para Object. Nem todo Object é 
uma String, então o compilador não vai deixar você fazer essa conversão. 
Lembre-se que, em geral, o compilador não conhece os valores das variáveis, 
apenas seus tipos. 


Vamos precisar moldar a referência para que o código compile: 


String recuperada = (String) objetos[0]; 


A partir de agora, esse código compila. Mas será que roda? Du- 
rante a execução, o casting vai ver se aquele objeto é mesmo compatível 
com o tipo String (no nosso caso é). Se não fosse, ele lançaria uma 





ClassCastException (exceção unchecked). 


Considere as classes: 


class Veiculo {} 
class Moto extends Veiculo (> 
class Carro extends Veiculo {} 
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E o código: 


Veiculo v = new Carro(); 
Moto m = v; 


Na primeira linha, usamos polimorfismo para chamar um Carro de 
Veiculo (é um). Na segunda linha, o que o compilador sabe é que v é do 
tipo Veiculo. E nem todo Veiculo é uma Moto. Por isso, essa linha não 
compila. 

Mas existem alguns Veiculo que são Moto. Então, o compilador deixa 
que façamos o casting: 


Veiculo v = new Carro(); 
Moto m = (Moto) v; 


Com isso, o código compila, mas repare que, em tempo de execução, 
v aponta para um objeto do tipo Carro. Quando o código for executado, 





haverá um erro de execução: ClassCastException. Carro não é uma 
Moto. 


Cuidado que, se o casting for totalmente impossível, o compilador já 
acusará erro: 


Carro c = new Carro(); 
Moto m = (Moto) c; 


Um Carro nunca poderá ser uma Moto. Então, nem com casting isso 
compila. 

É importante lembrar que quando não precisamos de casting, ele é op- 
cional, portanto todas as linhas a seguir funcionam com ou sem casting: 
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String guilherme = "guilherme"; 
String nome = guilherme; 


String nome2 = (String) guilherme; 


Object nome3 = guilherme; 


Object nome4 = (String) guilherme; 


Object nome5 = (Object) guilherme; 





REGRA GERAL! 


Se você está subindo na hierarquia de classes, a autopromoção vai 
fazer tudo sozinho; e se você estiver descendo, vai precisar de casting. 
Se não houver um caminho possível, não compila nem com casting. 

Na prova, faça sempre os diagramas de hierarquia de tipos que fica 
extremamente fácil resolver esses castings. 











Casting com interfaces 
Dado o código a seguir: 


Carro c = new Carro(); 
Moto m = (Moto) c; 


Quando dizemos que ele não compila, é porque um Carro nunca pode 
ser uma Moto. Mas como o compilador sabe que isso é impossível mesmo? 
Existe alguma chance de algum objeto de qualquer tipo ser, ao mesmo tempo, 


Carro e Moto? 


class X extends Moto, Carro { // não compila! 


A única maneira de isso acontecer seria se Java suportasse herança múlti- 
pla; aí escreveríamos uma classe que herdasse de Carro e Moto ao mesmo 
tempo. Como Java não tem herança múltipla, isso realmente é impossível de 
acontecer. 

Mas e quando fazemos casting com interfaces envolvidas? Apesar de não 
existir herança múltipla, podemos implementar múltiplas interfaces! Fazer 
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casting para interfaces sempre é possível e vai compilar (há apenas uma ex- 
ceção a essa regra). 

Pegue uma interface qualquer, por exemplo Runnable. O código a 
seguir compila: 


Carro c = new Carro(); 
Runnable r = (Runnable) c; 


Um Carro pode serum Runnable? Sabemos que a classe Carro pro- 
priamente não implementa essa interface. Mas existe a possibilidade de existir 
algum objeto em Java que seja, ao mesmo tempo, Carro e Runnable? 


A resposta é sim! E se tivéssemos uma classe CarroRodavel? 


class CarroRodavel extends Carro implements Runnable { ... + 


O compilador não sabe o valor da variável c nesse exemplo. Ele não sabe 
que na verdade é uma instância de Carro e não de CarroRodavel. Ele 
sabe apenas que é do tipo Carro e, pela simples possibilidade de existir um 
objeto que seja Carro e Runnable, ele deixa o código compilar. 

Mas repare que a classe CarroRodavel1 não existe no diagrama original. 
Mesmo assim, o código compila! Apenas com a possibilidade de existir uma 
classe dessa, o compilador já aceita aquele casting, mesmo que uma classe 
dessas não exista na prática. 


Claro que o objeto é do tipo Carro, que não implementa Runnable e, 





em tempo de execução, vai ocorrer uma ClassCastingException. 





E FINAL 


Dizemos que o código anterior compila porque há a possibilidade de 
uma classe como CarroRodavel existir algum dia. Mas será que sem- 
pre há essa possibilidade mesmo? 

Se a classe Carro for final, é impossível existir uma classe filha 
dela. E como a própria Carro não implementa Runnable, nesse caso, 
será impossível fazer o casting para Runnable (o próprio compilador já 
acusa erro). 
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DICA 


Muitos exercícios são sobre casting de referência. Uma dica é seguir 
o que é possível, impossível e óbvio. 

Se é óbvio que o casting funciona, isso é, se a conversão é sempre 
verdade, a autopromoção faz sozinha. 

Se o casting é possível, mas nem sempre é verdade, o casting compila, 
mas pode lançar erro em tempo de execução. 

Se o casting é impossível, isto é, ele nunca pode dar certo, o código 
não vai compilar nem com casting. 

Em alguns livros, você encontra tabelas complicadas e grandes que o 
“ajudam” a decidir se o casting compila e roda, mas é muito mais fácil 
seguir pela lógica. 











instanceof 


O operador instanceof (a instanceof Classe) devolve true 
caso a referência a aponte para um objeto compatível (assignable, atribuível) 
ao tipo Classe 


Object c = new Carro(); 
boolean b1 = c instanceof Carro; // true 
boolean b2 = c instanceof Moto; // false 


O instanceof não compila se a referência em questão for obviamente 
incompatível, por exemplo: 


String s = "a"; 


boolean b = s instanceof java.util.List; // não compila 
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DETALHE 


instanceof é um operador que deve ser usado com extremo 
cuidado no dia a dia. Em muitos casos, ele indica a fraca modelagem de 
º c . » º 
um sistema, com blocos que parecem “switchs” e poderiam ser trocados 
por polimorfismo. 








2) 


N 


w 


> 


v 


a 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


interface Z {} 
interface W {}; 
interface Y extends Z, W {} 
class B {} 
class C extends B implements Y {} 
class D extends B implements Z, W {} 
class E extends C {} 
class A { 
public static void main(String[] args) { 
B b = new C(); 


a) Não compila na definição das classes e interfaces. 
b) Não compila dentro do método main. 
c) Compila e roda sem exception. 


d) Compila e roda dando exception. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


interface Z {} 

interface W {} 

interface Y extends Z, W {} 

class B {} 

class C extends B implements Y {} 
class D extends B implements Z, W {} 
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class E extends C {} 


N 


s class A { 

9 public static void main(Stringl] args) 1 
10 C c = (C) new BC); 

nu } 

2 + 


a) Não compila na definição das classes e interfaces. 
b) Não compila dentro do método main. 
c) Compila e roda sem exception. 


d) Compila e roda dando exception. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


interface Z {} 

2 interface W {} 

interface Y extends Z, W {} 

class B {} 

s class C extends B implements Y {} 


n 


w 


ES 


é class D extends B implements Z, W {} 
7 class E extends C {} 


s class A { 

9 public static void main(Stringl] args) 1 
10 Y y = new D(); 

nu } 

2 + 


a) Não compila na definição das classes e interfaces. 
b) Não compila dentro do método main. 
c) Compila e roda sem exception. 


d) Compila e roda dando exception. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı interface Z {} 
2 interface W {} 
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Es 


v 


5) 


N 


v 


interface Y extends Z, W {} 
class B {} 
class C extends B implements Y {} 
class D extends B implements Z, W {} 
class E extends C {} 
class A { 
public static void main(String[] args) { 
Y y = (Y) new DO; 


a) Não compila na definição das classes e interfaces. 
b) Não compila dentro do método main. 
c) Compila e roda sem exception. 


d) Compila e roda dando exception. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


interface Z {} 
interface W {} 
interface Y extends Z, W {} 
class B {} 
class C extends B implements Y {} 
class D extends B implements Z, W (J 
class E extends C {} 
class A { 
public static void main(String[] args) { 
Z z= (Z) (B) new D(); 


a) Não compila na definição das classes e interfaces. 
b) Não compila dentro do método main. 
c) Compila e roda sem exception. 


d) Compila e roda dando exception. 
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6) 


7) 


n 


N 


v 


N 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


interface Z {} 
interface W {} 
interface Y extends Z, W {} 
class B {} 
class C extends B implements Y {} 
class D extends B implements Z, W {} 
class E extends C {} 
class A { 
public static void main(String[] args) { 
Y y= (Y) new AQ; 


a) Não compila na definição das classes e interfaces. 
b) Não compila dentro do método main. 
c) Compila e roda sem exception. 


d) Compila e roda dando exception. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


interface Z {} 

interface W {} 

interface Y extends Z, W {} 

class B {} 

class C extends B implements Y {} 

class D extends B implements Z, W {} 

class E extends C {} 

class A { 

public static void main(String[] args) { 

D d = (D) (Y) (B) new DO; 


a) Não compila na definição das classes e interfaces. 


b) Não compila dentro do método main. 
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8) 


9) 


N 


c) Compila e roda. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


interface Z {} 

interface W {} 

interface Y extends Z, W {} 

class B {} 

class C extends B implements Y {} 

class D extends B implements Z, W {} 

class E extends C {} 

class A { 
public static void main(String[] args) { 

System.out.printin(((B) (Z) (W) (Y) new D()) instance0f D); 

F 


a) Não compila na definição das classes e interfaces. 
b) Não compila dentro do método main. 
c) Compila e imprime true. 


d) Compila e imprime false. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


interface Z {} 

interface W {} 

interface Y extends Z, W {} 

class B {} 

class C extends B implements Y {} 

class D extends B implements Z, W {} 

class E extends C {} 

class A { 
public static void main(String[] args) { 

System.out.printin(((B) (Z) (W) (Y) new D()) instanceof D); 

} 


a) Não compila na definição das classes e interfaces. 
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b) Não compila dentro do método main. 
c) Compila e imprime true. 


d) Compila e imprime false. 


9.5 USE SUPER E THIS PARA ACESSAR OBJETOS E CON- 
STRUTORES 


Um construtor pode ser sobrecarregado assim como os métodos, e pode ter 
qualquer modificador de visibilidade. 

O ponto mais importante sobre os construtores é que, para construir um 
objeto de uma classe filha, obrigatoriamente, precisamos chamar um constru- 
tor da classe mãe antes. Sempre, em todos os casos. Para chamar o construtor 


da mãe, usamos a chamada ao super (passando ou não argumentos): 


class Mae { 
public Mae(String msg) { 
System.out.printlin(msg); 


class Filha extends Mae 1 
public Filha(String nome) { 
super ("construindo parte mae"); 
System.out.println('construindo parte filha"); 


Mas, na maioria dos casos, não chamamos o construtor da mãe ex- 
plicitamente. Se nenhum construtor da mãe foi escolhido através da 
palavra super (...), o compilador coloca automaticamente super (); 
no começo do nosso construtor, sem nem olhar para a classe mãe. 


class Mae 1 
public Mae() 1 
System.out.println('construindo parte mae"); 


308 


Casa do Código Capítulo 9. Trabalhando com herança 





class Filha extends Mae 1 
public Filha(String nome) { 
// super() esta implícito!!! 
System. out .println("construindo parte filha"); 


Considerando agora o código: 


public class X{ 
public static void main(String [] args){ 
Filha filha = new Filha("Teste"); 


} 
} 
Vai primeiro imprimir “Construindo parte mae” e só depois “Constru- 
indo parte filha”. 


Uma outra possibilidade, no caso de termos mais de um construtor, é 
chamarmos outro construtor da própria classe através do this (): 


class Mae { 
public Mae() 1 
System.out.printin('construindo parte mae"); 


class Filha extends Mae 1 
public Filha() 1 
// super() implicito! 
System.out.println('construindo filha parte 1"); 


public Filha(String nome) 1 
this(); 
System.out.println('construindo filha parte 2"); 
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public class X{ 
public static void main(String [] args) 
Filha filha = new Filha("Teste'"); 


» c 


Agora vai produzir “Construindo parte mae” “Construindo parte filha 
parte 1º e “Construindo parte filha parte 2”, 

Atenção, a chamada do construtor com super ou this só pode apare- 
cer como primeira instrução do construtor. Portanto, só podemos fazer uma 
chamada desses tipos. 


class Filha extends Object { 
public Filha() { 
// super() implicito! 


public Filha(String nome) 1 
this(D); 


public Filha(int idade) { 
super (); 
this(); // não compila, ou um ou outro! 


} 
public Filha(long valor) { 

this(); 

this(); // não compila, só uma vez! 
} 


public Filha(char caracter) { 
super (); 
super(); // não compila, só uma vez! 
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this e variáveis membro 


Por vezes, temos variáveis membro com o mesmo nome de variáveis lo- 
cais. O acesso sempre será a variável local, exceto quando colocamos o this, 
que indica que a variável membro será acessada. O código a seguir imprimirá 
3e depois 5: 


class Teste { 

int i = 5; 

void roda(int i) 1 
System.out.printin(i); 
System.out.println(this.i); 

J 

public static void main() { 
new Teste().roda(3); 


No acesso a variáveis membro com o this podem parecer que serão 
acessados somente valores da classe atual, mas buscam também nas classes 
da qual ela herda: 


class AÍ 
int i = 5; 
} 
class Teste extends A{ 
void roda(int i) { 
System.out.println(this.i); // imprime 5 
} 
public static void main() { 
new Teste().roda(3); 


Tentar acessar uma variável local com this não compila: 


class Teste { 
void roda(int i) { 
System.out.println(this.i); // não há variável membro i 
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public static void main() 1 
new Teste() .roda(3); 


Como mostramos, caso a variável seja escondida por uma variável com 
mesmo nome em uma classe filha, podemos diferenciar o acesso à variável 
membro da classe filha ou da pai, explicitando this ou super: 


class M 
int i = 5; 
} 
class Teste extends Aí 
int i = 10; 
void roda(int i) { 
System.out.println(i); // imprime 3 
System.out.println(this.i); // imprime 10 
System.out.println(super.i); // imprime 5 
} 
public static void main() { 
new Teste().roda(3); 


O this é em geral opcional para acessar um método do nosso objeto 
atual (se ele não foi redefinido, da classe mãe): 


class AÍ 
int i() { return 5; +) 
} 
class Teste extends AÍ 
void roda() { 
System.out.println(this.i()); // imprime 5 
} 
public static void main() { 
new Teste().roda(); 


} 
class Teste2 { 


312 


Casa do Código Capítulo 9. Trabalhando com herança 





int i() { return 5; } 
void roda() 1 
System.out.println(this.i()); // imprime 5 
} 
public static void main() { 
new Teste().roda(); 


this e super em variável membro 


E o que acontece quando uma variável membro tem o mesmo nome que a 
definida na classe que herdamos? Se não definirmos o acesso através de this 
nem super, o acesso é à variável da classe filha. Se usarmos this é à classe 
filha novamente e se usarmos super é à classe pai: 


class Veiculo 1 
double velocidade = 30; 
} 
class Carro extends Veiculo { 
double velocidade = 50; 
void imprime() { 
System.out.println(velocidade); // 50 
System.out.println(this.velocidade); // 50 
System.out.println(super.velocidade); // 30 


} 
class Teste { 
public static void main(String[] args) { 
Carro c = new Carro(); 
c.imprime(); 


Lembre-se que o binding de uma variável ao tipo é feito em compi- 
lação, portanto se tentarmos acessar a variável velocidade fora do Carro 
através de uma referência a Carro, o valor alterado é o da variável 


Carro.velocidade 
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class Veiculo { 
double velocidade = 30; 
} 
class Carro extends Veiculo { 
double velocidade = 50; 
void imprime() { 
System.out.println(velocidade); // 1000 
System.out.println(this.velocidade); // 1000 
System. out .println(super.velocidade); // 30 


} 
class Teste { 
public static void main(String[] args) { 
Carro c = new Carro(); 
c.velocidade = 1000; 
c.imprime(); 


E se fizermos o mesmo através de uma referência a Veiculo, alteramos 
a velocidade do Veiculo: 


class Veiculo 1 
double velocidade = 30; 
} 
class Carro extends Veiculo { 
double velocidade = 50; 
void imprime() { 
System.out.println(velocidade); // 50 
System. out .println(this.velocidade); // 50 
System.out .println(super.velocidade); // 1000 


} 
class Teste { 
public static void main(String[] args) { 
Carro c = new Carro(); 
( (Veiculo) c).velocidade = 1000; 
c.imprime(); 
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Estático não tem this nem super 


Contextos estáticos não possuem nem this nem super, uma vez que 
o código não é executado dentro de um objeto: 


class M 
int i = 5; 
} 
class Teste extends AÍ 
int i = 10; 
public static void main() 1 
this.i = 5; // this? não compila. código estático 
super.i = 10; // super? não compila. código estático 
} 


Por fim, uma última restrição: interfaces não podem ter métodos estáti- 
cos, não compila (métodos default não são cobrados nesta prova). 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { 

2 int x = 1; 

3 5 

4 class A extends B { 

5 static int x = 2; 

6 public static void main(String[] args) { 
7 System.out.printin(x); 

8 J 

9 5 


a) Não compila. 
b) Compila e imprime. 
c) Compila e imprime 2. 


d) Compila e dá exception. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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3) 


1 


4) 


i 


2 


class B { 
int x = 1; 
} 
class A extends B { 
static int x = 2; 
public static void main(String[] args) { 
System. out .println(this.x); 


a) Não compila. 
b) Compila e imprime 1. 
c) Compila e imprime 2. 


d) Compila e dá exception. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B { 
int x = 1; 
} 
class A extends B { 
static int x = 2; 
public static void main(String[] args) { 
System.out.printin(super.x); 


a) Não compila. 
b) Compila e imprime. 
c) Compila e imprime 2. 


d) Compila e dá exception. 


Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


class B { 
int x = 1; 
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3 5 

4 class A extends B { 

5 static int x = 2; 

6 public static void main(Stringl] args) 1 
7 System.out.printin(new A() .super.x); 
8 } 

ə} 


a) Não compila. 
b) Compila e imprime 1. 
c) Compila e imprime 2. 


d) Compila e dá exception. 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { 

2 void BO { 

3 + 

4 void B(String s) É 
5 this(); 

6 this(s); 

7 } 

8 + 

ə class A { 

10 public static void main(Stringl] args) 1 
u B b = new B(); 
12 } 

3 } 


a) Não compila. 
b) Compila e dá exception. 
c) Compila e não imprime nada. 


d) Compila e entra em loop infinito. 


6) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 
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ı class B { 

2 BO { 

3 } 

P B(String s) { 

5 this(); 

6 this(s); 

7 } 

8 + 

ə class A { 

10 public static void main(Stringl] args) 1 
u B b = new B(); 
12 } 

5 } 


a) Não compila. 
b) Compila e dá exception. 
c) Compila e não imprime nada. 


d) Compila e entra em loop infinito. 


7) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { 

2 BO { 

3 } 

á B(String s) { 

5 this(); 

6 } 

7} 

s class A { 

9 public static void main(Stringl] args) 1 
10 B b = new B(); 
nu } 

2 + 


a) Não compila. 
b) Compila e dá exception. 


c) Compila e não imprime nada. 
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d) Compila e entra em loop infinito. 


8) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { 

2 BO + 

3 } 

4 B(String s) { 

5 this(); 

6 bi 

75 

s class A { 

9 public static void main(Stringl] args) 1 
10 String s = null; 
u B b = new B(s); 
12 } 

5 5 


a) Não compila. 
b) Compila e dá exception. 
c) Compila e não imprime nada. 


d) Compila e entra em loop infinito. 


9) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { 

2 int x() { return y();} 
3 int yO { return 3; + 
4 + 

s class C extends B { 

6 cO í 

7 this(x()); 

8 } 

5 Clint i) 1 

10 System.out.printin(i); 
11 } 

iż int y() { return 2; } 
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5 5 

4 class A { 

15 public static void main(String[] args) { 
16 new C(); 

17 } 

18 + 


a) Não compila. 
b) Compila e imprime ‘25 


c) Compila e imprime 3. 


10) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class B { 

à int x() { return y();} 

3 int y() { return 3; } 

4} 

s class C extends B { 

6 cO { 

a super(); 

8 z(x0); 

9 } 

10 void z(int i) { 

u System.out.println(i); 
12 } 

13 int y() { return 2; } 

14 } 

5 class A { 

16 public static void main(String[] args) { 
17 new C(); 

18 } 

19 5 


a) Não compila. 
b) Compila e imprime ‘25 


c) Compila e imprime “3 
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9.6 USE CLASSES ABSTRATAS E INTERFACES 


Classes e métodos podem ser abstratos. 
Uma classe abstrata pode não ter nenhum método abstrato: 


// compila 
abstract class SemMetodos { 


} 


Se uma classe tem um método que é abstrato, ela deve ser declarada como 
abstrata, ou não compilará. 


// não compila, se tem método abstrato, tem que implementar 
class ComMetodoAbstrato { 
public abstract void executa(); 


Uma classe abstrata não pode ser instanciada diretamente: 


abstract class X{ 


public class Teste{ 
public static void main(String[] args) { 
X x = new X(); 


A classe Teste não compila! Classes abstratas não podem ser instanci- 
adas diretamente: 


Teste. java:7: X is abstract; cannot be instantiated 
X x = new X(); 


1 error 


Um método abstrato é um método sem corpo, somente com a definição. 
Uma classe que tem um ou mais métodos abstratos precisa ser declarada como 
abstrata. 

Não importa se o método foi escrito diretamente ou foi herdado: 
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abstract class Veiculo { 
public abstract void liga(); 


// compila pois implementou 
class Moto extends Veiculo { 
public void liga() 1 


// compila pois a classe é abstrata, com método herdado 
// abstrato ainda 
abstract class QuatroRodas extends Veiculo { 


} 


// não compila pois a classe não é abstrata, 
// com método herdado abstrato ainda 
class SemRodas extends Veiculo { 


} 


Um método abstrato tem de ser reescrito ou herdado pelas suas filhas 
concretas. 


Agora veja o exemplo a seguir: 


abstract class Veiculo { 
public abstract void liga(); 


class Moto extends Veiculo { 
public void liga() { 


O método liga foi implementado na classe filha, então ela pode ser 
concreta. Basta pensar que métodos abstratos herdados são responsabilidades 
herdadas: você não poderá ser um objeto concreto enquanto tiver respons- 
abilidades a serem tratadas. 
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Quando herdamos de uma classe abstrata que possui um método abstrato, 
temos que escolher: ou implementamos o método, ou somos abstratos tam- 
bém e passamos adiante a responsabilidade. Note que a classe pode imple- 
mentar o método e mesmo assim também ser abstrata. 


abstract class Veiculo 1 
public abstract void liga(); 


abstract class Moto extends Veiculo { 


class MotoEspecial extends Moto 1 
public void liga() { 


// compila: decidi implementar mas mesmo assim 
// manter a classe abstrata 
abstract class QuatroRodar extends Veiculo { 
public void liga() { 
} 


Código de uma classe abstrata pode acessar o código da classe concreta, 
uma vez que ele só será executado quando o objeto for criado: 


abstract class X{ 
abstract void x() { 
System. out.println(y()); 
} 
abstract String y(); 
} 
class Y extends X { 
String yO { 
return "codigo"; 
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} 
public class Teste { 
public static void main(String[] args) { 
new X().x(); // imprime código 


Interfaces 


Uma interface declara métodos que deverão ser implementados pelas 
classes concretas que queiram ser consideradas como tal. Por padrão, são 
todos métodos públicos e abstratos. 


interface Veiculo { 
void ligar(); 
// public abstract! Você pode escrever, mas é por padrão 
// isso. 
public abstract int pegaMarcha(); 


Quando você implementa a interface em uma classe concreta, é preciso 
implementar todos os métodos. Similarmente, ao herdar uma classe abstrata, 
a classe concreta deve implementar todos os métodos que não foram imple- 
mentados ainda: 


// compila, todos os métodos implementados 
class Carro implements Veiculo { 
public void ligar) { 
} 
public int pegaMarcha() { 
return 0; 


} 
// não compila, onde está o pegaMarcha? 
class Moto implements Veiculo { 
public void ligar() { 
hi 
} 


// não compila, o método pegaMarcha definiu escopo default, 
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// quando deveria definir public 
class Triciclo implements Veiculo 1 
public void ligar) + 
} 
int pegaMarcha() { 
return 0; 


Valem as mesmas regras de quando você herda de uma classe abstrata: ou 
você tem todos os métodos reescritos, e aí pode declará-la como concreta, ou 
então você precisa declará-la como abstrata. 


// compila, pois a classe é abstrata 
abstract class Moto implements Veiculo { 
public void ligar() { 
} 


Uma classe pode implementar diversas interfaces: 


abstract class MinhaClasse implements Serializable, Runnable { 


} 


Justamente por isso, a prova vê como um bom uso de interfaces quando 
você quer herdar de dois lugares mas a herança de classes não permite. Para 
a prova, essa razão é suficiente, mas na prática existe uma diferença grande 
entre composição (herança de interfaces não envolve herdar comportamento 
e variáveis membro) e herdar comportamento e variáveis membro de uma 
classe mãe. Como a implementação de uma interface nos obriga a escrever 
todos os métodos, estamos compondo nossa classe de diversas interfaces. 

Lembre-se que uma interface pode herdar de outra, inclusive de diversas 
interfaces: 


interface A extends Runnable {} 
interface B extends Serializable {} 
interface C extends Runnable, Serializable {} 


Note que uma interface nunca implementa outra interface: 
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interface A implements Runnable {} // não compila 


Você pode declarar variáveis em uma interface, todas elas serão public 
final static, isto é, constantes. 
interface X { 

int i = 5; 

// você até pode escrever public static final, mas é sempre 


// assim 


Uma interface, por sua vez, pode estender outra interface, herdando suas 
responsabilidades e constantes. Uma interface pode estender mais de uma 


interface! 


interface X extends Runnable, Comparable { } 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı abstract class B { 


2 void x() { 

3 System.out.printin(y (0); 
4 } 

E abstract int yO; 

6 5 

7 abstract class C extends B { 

8 int yO { return 1; ) 

9 + 

1 class D extends C { 

i int y() { return 2; } 

12 + 

3 class A { 

14 public static void main(String[] args) { 
15 Dd = (D) (C) new DC); 
16 d.x(); 

17 } 

18 + 


a) Não compila. 
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b) Compila e imprime “1. 
c) Compila e imprime ‘25 
d) Compila e roda com exception. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı abstract class B { 


2 abstract void x() { 

; System.out.printin(y 0); 
4 } 

5 abstract int y(); 

6} 

7 abstract class C extends B { 

8 int yO { return 1; + 

ə 5 


io class D extends C { 
n int y() { return 2; } 


n2 } 

3 class A { 

14 public static void main(String[] args) { 
15 Dda = (D) (C) new D(; 

16 d.x(); 

17 F} 

18 + 


a) Não compila. 
b) Compila e imprime “í. 
c) Compila e imprime ‘25 


d) Compila e roda com exception. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı abstract class B { 


2 void x() { 
3 System.out.println(y()); 
4 } 
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A abstract int y(); 

6} 

7 abstract class C extends B { 

8 abstract int yO; 

9 5 

1 class D extends C 1 

u int yO { return 1; ) 

12 + 

3 class A { 

14 public static void main(Stringl] args) 1 
15 Da = (D) (C) new DC); 
16 d.x(); 

17 } 

18 + 


a) Não compila. 
b) Compila e imprime “1$ 
c) Compila e imprime ‘2. 


d) Compila e roda com exception. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı abstract class B { 


2 void x() { 

3 System. out .println(y()); 
4 } 

5 int yO | 

6 return 2; 

7 F 

s 5 

ə abstract class C extends B { 

10 abstract int y(); 

11 } 

» class D extends C { 

3 int y() { return 1; } 

14 } 

5 class A { 

16 public static void main(String[] args) { 
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17 Dad = (D) (C) new D(; 
18 d.x(); 

19 } 

20 + 


a) Não compila. 
b) Compila e imprime “1. 
c) Compila e imprime ‘25 


d) Compila e roda com exception. 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı abstract class B { 


2 void x() { 

; System.out.println(y()); 
4 } 

; final int y() { 

6 return 2; 

7 } 

s 5 


ə abstract class C extends B { 
10 int yO { 


u return 3; 

12 } 

5 5 

4 class D extends C { 

15 int y() { return 1; } 

16 + 

7 class A { 

18 public static void main(Stringl] args) 1 
19 Da = (D) (C) new D(; 
20 d.x(); 

21 } 

2 + 


a) Não compila. 


b) Compila e imprime ‘1$ 
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c) Compila e imprime “2 


d) Compila e roda com exception. 


6) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı abstract class B { 


2 void x() { 

3 System.out.printin(y()); 
4 } 

5 Object y() { return "a"; 5 

6 + 

7 abstract class C extends B 1 

8 abstract String yO; 

ə } 

1 class D extends C { 

u String yO { return "b"; 5 
2 } 

3 Class A { 

14 public static void main(Stringl] args) 1 
15 Dd = (D) (C) new DC); 
16 d.x(); 

17 } 

18 + 


a) Não compila. 
b) Compila e imprime ʻa‘ 
c) Compila e imprime bi. 


d) Compila e roda com exception. 
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Lidando com exceções 


10.1 DIFERENCIE ENTRE EXCEÇÕES DO TIPO 
CHECKED, RUNTIME E ERROS 


Durante a execução de uma aplicação, erros podem acontecer. A lin- 
guagem Java oferece um mecanismo para que o programador possa definir 
as providências apropriadas a serem tomadas na hora em que um erro de ex- 
ecução ocorrer. 

Os erros de execução são classificados em algumas categorias. É funda- 
mental que você seja capaz de, dado um erro de execução, determinar seu 
tipo. A classificação das categorias depende exclusivamente da hierarquia das 
classes que modelam os erros de execução. 

A classe principal dessa hierarquia é a Throwable. Qualquer erro de 
execução é um objeto dessa classe ou de uma que deriva dela. 


10.1. Diferencie entre exceções do tipo checked, runtime e erros Casa do Código 











Como filhas diretas de Throwable temos: Error e Exception. 





Os Errors são erros de execução gerados por uma situação total- 
mente anormal que não deveria ser prevista pela aplicação. Por exemplo, 





um OutOfMemoryError é gerado quando a JVM não tem mais memória 
RAM disponível para oferecer para as aplicações. Em geral, esse tipo de erro 
não é responsabilidade das aplicações pois quem cuida do gerenciamento de 
memória é a JVM. 





Por outro lado, as Exceptions são erros de execução que são de respon- 
sabilidade das aplicações, ou seja, são as aplicações que devem tratar ou evitar 





esses erros. Por exemplo, um SQLException é gerado quando algum erro 
ocorre na comunicação entre a aplicação e o banco de dados. Esse tipo de 
erro deve ser tratado ou evitado pela aplicação. 





Por sua vez, as Exceptions são divididas em duas categorias: as 
uncheckeds e as checkeds. As uncheckeds são exceptions que teoricamente 
podem ser mais facilmente evitadas pelo próprio programador se ele codi- 
ficar de maneira mais cuidadosa. As checkeds são exceptions que teoricamente 
não são fáceis de evitar, de modo que a melhor abordagem é estar sempre 
preparado para seu acontecimento. 

As uncheckeds são definidas pelas classes que derivam de 








RuntimeException, que por sua vez é filha direta de Exception. 





As outras classes na árvore da Exception definem as checkeds. 


Throwable 
Pai i 
Exception Error 
FA SN. 
RuntimeException I0Exception 
NS a 
NullPointerException ClassCastException FileNotFoundException 


ArraylndexOutOfBoundsException 
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Essas diferenças nao ficam apenas na teoria. O compilador irá verificar se 
seu programa pode lançar alguma checked exception e, neste caso, obrigá-lo a 
tratar essa exception de alguma maneira. No caso das exceptions unchecked, 
não há nenhuma verificação por parte do compilador pelo tratamento ou não. 


1) Dentre as classes a seguir qual delas não é checked? 


e java.io. IOException 





e java. sql.SQLException 





e java. lang.Exception 





e java. lang. IndexOutOfBoundsException 














e java.io.FileNotFoundException 


10.2 DESCREVA O QUE SÃO EXCEÇÕES E PARA QUE SÃO 
UTILIZADAS EM JAVA 


Imagine a situação em que tentamos acessar uma posição em um array: 


public void fazAlgo(int[] idades) 1 
System.out.printin(idades[0]); 


O que acontece se o array enviado para o método é vazio? Se esse código 
imprimisse nulo ou um número padrão, nesse caso, teríamos sempre que nos 


preocupar, como em: 


public void fazAlgo(int[] idades) 1 
if (idades [0]==null) return; 
// return para caso o Java devolva nulo ao acessar 
// uma posição inválida 


System.out.printin(idades[0]); 
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Pense como seria difícil tratar todas as situações possíveis que fogem do 
padrão de comportamento que estamos desejando. Nesse caso, o comporta- 
mento padrão, aquilo que acontece 99% das vezes e que esperamos que acon- 
teça é que a posição acessada dentro do array seja válido. Não queremos ter 
que verificar toda vez se o valor é válido, e não queremos entupir nosso código 
com diversos ifs para diversas condições. As exceções à regra, as exceptions, 
são a alternativa para o controle de fluxo: em vez de usarmos ifs para con- 
trolar o fluxo que foge do padrão, é possível usar as exceptions para esse papel. 
Veremos adiante como tratar erros, como o acesso a posições inválidas, tentar 
acessar variáveis com valores inválidos etc. 

Caso uma exception estoure e sua stack trace seja impressa, teremos algo 


como: 


Exception in thread "main" 

java. lang.ArrayIndexQut0fBoundsException: O 
at SuaClasse.fazAlgo (SuaClasse. java:20) 
at SuaClasse.main(SuaClasse.java:30) 


Note como a stack trace indica que método estava sendo invocado, em 
qual linha do arquivo fonte está essa invocação, quem invocou este método 
etc. 

O importante é lembrar que as exceptions permitem que isolemos o trata- 
mento de um comportamento por blocos, separando o bloco de lógica de 
nosso negócio do bloco de tratamentos de erros (sejam eles Exceptions ou 





Errors, como veremos adiante). O stack trace de uma Exception também 
ajuda a encontrar onde exatamente o problema ocorreu e o que estava sendo 
executado naquela Thread naquele instante. 

1) Escolha 2 opções. 


Exception é um mecanismo para... 


e tratar entrada de dados do usuário. 


* que você pode usar para determinar o que fazer quando algo ines- 
perado acontece. 


e que a VM usa para fechar o programa caso algo inesperado aconteça. 
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e controlar o fluxo da aplicação. 


* separar o tratamento de erros da lógica principal. 


2) De que maneira a API de exceptions pode ajudar a melhorar o código de 


seu programa? 


Escolha 2 opções: 


10.3 


Permitindo separar o tratamento de erro da lógica do programa. 
Permitindo tratar o erro no mesmo ponto onde ele ocorre. 


Permitindo estender as classes que já existem e criar novas excep- 


tions. 


Disponibilizando várias classes com todas as exceptions possíveis 
prontas. 


Aumentando a segurança da aplicação disponibilizando os erros nos 
logs. 


CRIE UM BLOCO TRY-CATCH E DETERMINE COMO 
EXCEÇÕES ALTERAM O FLUXO NORMAL DE UM 
PROGRAMA 


O programador pode definir um tratamento para qualquer tipo de erro de 


execução. Antes de definir o tratamento, propriamente, é necessário deter- 


minar o trecho de código que pode gerar um erro na execução. Isso tudo é 


feito com o comando try-catch. 


try É 


// trecho que pode gerar um erro na execução. 


} catch (Throwable t) { // pegando todos os possíveis erros de 


//execução. 


// tratamento para o possível erro de execução. 
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A sintaxe do try-catch tem um bloco para o programador definir o 
trecho de código que pode gerar um erro de execução. Esse bloco é determi- 
nado pela palavra try. O programador também pode definir quais tipos de 
erro ele quer pegar para tratar. Isso é determinado pelo argumento do catch. 
Por fim, o tratamento é definido pelo bloco que é colocado após o argumento 
do catch. 

Durante a execução, se um erro acontecer, a JVM redireciona o fluxo de 
execução da linha do bloco do try que gerou o erro para o bloco do catch. 
Importante! As linhas do bloco do try abaixo daquela que gerou o erro não 
serão executadas. 


Fazer um catch em Throwable não é uma boa prática, pois todos os 





erros possíveis são tratados pela aplicação. Porém, os Errors não deveriam 


ser tratados pela aplicação, já que são de responsabilidade da JVM. Assim, 





também não é boa prática dar catch em Errors. 
Modificando o argumento do catch, o programador define quais erros 
devem ser pegos para serem tratados. 


try É 
// trecho que pode gerar um erro na execução. 

) catch (Exception e) { // pegando todas as exceptions. 
// tratamento para o possível erro de execução. 


Para a prova, é fundamental saber quando o programador pode ou não 
pode usar o try-catch. A única restrição de uso do try-catch en- 
volve as checked exceptions. Qual é a regra? O programador só pode usar 
try-catch em uma checked exception se o código do bloco do try pode 
realmente lançar a checked exception em questão. 


try É 

System.out.printin('não acontece SQLException"); 
+ catch(SQLException e){ // pegando SQLException. 

// tratamento. 


Esse código não compila pois o trecho envolvido no bloco do try nunca 





geraria a checked SQLException. O compilador avisa com um erro de 
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“unreachable code”. Já o exemplo a seguir compila, pois pode ocorrer um 





FileNotFoundException: 


try É 
new java.io.FileInputStream("a.txt'); 
+ catch(java.io.FileNotFoundException e){ 
// tratamento. 


O código a seguir não tem nenhum problema, pois o programador pode 
usaro try-catch em qualquer situação para os erros de execução que não 
são checked exceptions. 


try É 
System.out.printin("0k'); 
} catch (RuntimeException e) { // pegando RuntimeException 
// Cunckecked). 
// tratamento. 


Podemos pegar tudo, exceptions e erros: 


try É 
System.out.printin("0k'); 
} catch (Throwable e) { 
// tratamento 


Quando a exception é pega, o fluxo do programa é sair do bloco try 
e entrar no bloco catch, portanto, o código a seguir imprime peguei e 


continuando normal: 


String nome = null; 
try { 
nome .toLowerCase(); 
System. out .println("segunda linha do try"); 
} catch(NullPointerException ex) { 
System. out .println("peguei"); 
} 


System. out .println("continuando normal"); 
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Mas, se a exception que ocorre não é a que foi definida no catch, a 
chamada do método para e volta, jogando a exception como se não houvesse 
um try/catch. O cenário a seguir demonstra essa situação e não imprime 
nada: 


String nome = null; 

try É 
nome.toLowerCase () ; 
System.out.println("segunda linha do try"); 

+ catch(IndexOut0fBoundsException ex) { 
System.out.println("peguei"); 

} 


System.out.println('continuando normal"); 





Lembre-se sempre do polimorfismo, portanto, pegar IOException é 





o mesmo que pegar todas as filhas de I0Exception também. O código a 
seguir trata o caso de o arquivo não existir além de todas as outras filhas de 





IOException: 


try É 

new java.io.FileInputStream("a.txt'); 
+ catch(java. io. IDException e){ 

// tratamento. 


Bloco finally 


Tem coisas que não podemos deixar de fazer em hipótese alguma. Seja 
no sucesso ou no fracasso, temos obrigação de cumprir com algumas tarefas. 
Imagine um método que conecta com um banco de dados. Não im- 
porta o que aconteça, no fim desse método a conexão deveria ser fechada. 
Durante a comunicação com o banco de dados, há o risco de ocorrer uma 





SQLException 


void metodo()f 
try { 
abreConexao(); 
fazConsultas(); 
fechaConexao() ; 
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} catch (SQLException e) { 
// tratamento 





Nesse código, há um grande problema: se um SQLException ocorrer 
durante as consultas, a conexão com o banco de dados não será fechada. Para 
tentar resolver esse problema, o bloco do catch poderia invocar o método 





fechaConexao (). Então, se acontecesse um SQLException o bloco do 
catch seria executado e, consequentemente, a conexão seria fechada. 


Mas ainda não solucionamos o problema, pois outro tipo de erro poderia 








acontecer nas consultas. Por exemplo, uma NullPointerException que 
não está sendo tratada. Para resolver o problema de fechar a conexão, um 
outro recurso do Java será utilizado, o bloco finally. Esse bloco é sempre 
executado, tanto no sucesso quanto no fracasso por qualquer tipo de erro. 


void metodo(){ 
try { 
abreConexao () ; 
fazConsultas(); // Não precisa mais fechar a conexao 
// aqui. 
+ catch(SQLException e) { 
// tratamento 
) finally { 
fechaConexao(); // fechando a conexao no sucesso ou no 
// fracasso. 


Para melhor entender o fluxo do try-cacth como finally, veja o 


próximo exemplo. 


class A { 
void metodo() 1 
tryd 
//A 
//B 
kcatch(SQLException e){ 
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//C 
}finally{ 
//D 


e Em uma execução normal, sem erros nem exceções, ele executaria A, 
B, D, E. 








e Com SQLException em A, ele executaria C, D, 





eal 








e Com NullPointerException em A, ele executaria apenas D e 
sairia. 


e Se Afosseum System.exit (0);, ele apenas executa A e encerra o 
programa. 


e Se ocorresse um erro A, executaria apenas D (dependendo do erro). 


Uma outra maneira um pouco menos convencional de usar o finally 
é sem o bloco catch, como no exemplo a seguir. 


class M 
void metodo() { 
try É 
System.out.printin("imprime algo"); 
) finally { 
// sempre permite fechar 


} 


1) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 
2 public static void main(String[] args) { 
3 String nome; 
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à try {d 

5 nome. toLowerCase(); 

6 System.out.println("a"); 

7 } catch(NullPointerException ex) { 
8 System.out.printin("b'); 

9 } 

10 System.out.println("c"); 

11 } 

n2 + 


a) Não compila. 

b) Compila e, ao rodar, imprime “abc”. 

c) Compila e, ao rodar, imprime “be”. 
co» 


d) Compila e, ao rodar, imprime “a”. 


e) Compila e, ao rodar, imprime “b”. 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(String[] args) { 
3 String nome = null; 

3 try {í 

5 nome. toLowerCase(); 

6 System.out.println("a"); 

7 } catch(NullPointerException ex) { 
8 System.out.printin("b"); 

9 } 

10 System.out.println("c"); 

11 } 

n2 } 


a) Não compila.. 

b) Compila e, ao rodar, imprime “abc”. 
c) Compila e, ao rodar, imprime “be”. 
d) Compila e, ao rodar, imprime “a” 


e) Compila e, ao rodar, imprime “b”. 
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3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 
3 String nome; 

y try { 

5 nome .toLowerCase(); 

6 System.out.printin("a"); 

7 + catch(NullPointerException ex) { 
8 System.out.printin("b''); 

9 ) finally 1 

10 System.out.printin('c''); 

u } 

12 System. out.println("d"); 

13 } 

14 > 


a) Não compila. 
b) Compila e, ao rodar, imprime “abcd”. 
c) Compila e, ao rodar, imprime “bcd”. 
d) Compila e, ao rodar, imprime “ac”. 

p p 
e) Compila e, ao rodar, imprime “bc”, 
f) Compila e, ao rodar, imprime “ad”, 


g) Compila e, ao rodar, imprime “bd”. 


10.4 INVOQUE UM MÉTODO QUE JOGA UMA EXCEÇÃO 


Eventualmente, um método qualquer não tem condição de tratar um deter- 
minado erro de execução. Nesse caso, esse método pode deixar passar o erro 
para o próximo método na pilha de execução. 

Para deixar passar qualquer erro de execução que não seja uma checked 
exception, é muito simples: basta não fazer nada. 


class Teste { 
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void primeiro()1 
System. out .println("primeiro antes"); 
this.segundo(); 
System. out .println("primeiro depois"); 


void segundo() { 
String s = null; 
System. out .println("segundo antes"); 
s.length(); 
System. out .println("segundo depois"); 


O segundo método declara uma variável não primitiva e a inicializa 
com null. Logo em seguida, ele utiliza o operador . em uma refer- 
ência que sabemos estar nula. Nesse ponto, na hora da execução, um 








NullPointerException é gerado. Perceba que não há try-catch no 
segundo método, então ele não está pegando e tratando o erro, mas sim 
deixando-o passar. O primeiro método não define o try-catch, ou seja, 





também deixa passar o NullPointerException. O resultado é a im- 





pressão de primeiro antes, segundo antes. 

Agora, para deixar passar uma checked exception, o método é obrigado 
a deixar explícito (avisado) que pretende deixar passar. Na assinatura do 
método, o programador pode deixar avisado que pretende deixar passar de- 
terminados erros de execução. Isso é feito através da palavra-chave throws. 


class Teste { 


void primeiro(){ 

try É 
System.out.printin("primeiro antes"); 
this.segundo(); 
System.out.println("primeiro depois"); 

+ catch(IDException e) { 
// tratamento. 
System.out.println("primeiro catch"); 


343 


10.4. Invoque um método que joga uma exceção Casa do Código 





System.out.println("primeiro fim"); 


void segundo() throws IOException 1 
System.out.println("segundo antes"); 
System.in.read(); // pode lançar IOException 
System.out.println('segundo depois"); 


O segundo método invoca o read() no System.in. Essa invocação 





pode gerar um IOException, de modo que o segundo método tem duas 
alternativas: ou pega e trata o possível erro ou o deixa passar. Para deixar 
passar, o comando throws deve ser utilizado na sua assinatura do segundo 





método. Isso indicará que um IOException pode ser lançado. 
Dessa forma, o primeiro método que invoca o segundo pode receber uma 
IOException. Então, ele também tem duas escolhas: ou pega e trata usando 





try-catch, ou deixa passar usando o throws. O resultado é a impressão 
de primeiro antes, segundo antes, primeiro catche primeiro 


fim. 


Gerando um erro de execução 


Qualquer método, ao identificar uma situação errada, pode criar um erro 
de execução e lançar para quem o chamou. Vale lembrar que os erros de ex- 
ecução são representados por objetos criados a partir de alguma classe da 
hierarquia da classe Throwable, logo, basta o método instanciar um objeto 
de qualquer uma dessas classes e depois lançá-lo. 

Se o erro não for uma checked exception, basta criar o objeto e utilizar o 
comando throw para lançá-lo na pilha de execução (não confunda com o 


throws): 


class Teste { 
void primeiro (1 


try { 
this.segundo(); 
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+ catch (RuntimeException e) 1 
// tratamento. 


void segundo() 1 
throw new RuntimeException(); 


Se o erro for uma checked exception, é necessário também declarar na 
assinatura do método o comando throws: 


class Teste { 


void primeiro()1 
try É 
this.segundo() ; 
+ catch(Exception e) { 
// tratamento. 


void segundo() throws Exception 1 
throw new Exception(); 


Podemos ainda criar nossas próprias exceções, bastando criar uma classe 
que entre na hierarquia de Throwable. 


class MinhaException extends Exceptiont> 


Em qualquer lugar do código, é opcional o uso do try e catch de uma 
unchecked exception para compilar o código. Em uma checked exception, é 
obrigatório o uso do try/catchou throws. 

O exemplo a seguir mostra uma unchecked exception sendo ignorada e 
o erro vazando, e nada será impresso: 
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public class Teste { 


public static void main(Stringl] args) 1 
metodo () ; 
System.out.printin("Apos a invocacao do metodo"); 


private static void metodo() 1 
int[] i= new int[10]; 
System.out.println(i[15]); 
System.out.println("Apos a exception"); 


Ao pegarmos a exception, será impresso também “Apos a invocacao do 
metodo” uma vez que após o catch, o fluxo volta ao normall: 


public class Teste 1 


public static void main(Stringl] args) 1 
try { 
metodo () ; 
+ catch (RuntimeException ex) { 
System.out.println("Exception pega"); 
} 


System.out.println("Apos a invocacao do metodo"); 


private static void metodo() { 
int[] i= new int[10]; 
System.out.println(i[15]); 
System.out.printin('"Apos a exception"); 


Podemos ter também múltiplas expressões do tipo catch. Nesse caso, 
será invocada somente a cláusula adequada, e não as outras. No código a 
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seguir, seo metodo2 jogar uma Array IndexOutOfBoundsException, 
será impresso runtime: 


void metodoi() 1 

try É 
metodo2(); 

+ catch(IDException ex) { 
System.out.printin("io''); 

+ catch(RuntimeException ex) { 
System.out.printlin(''runtime"); 

+ catch(Exception ex) { 
System. out .println("exception qualquer"); 


E a ordem faz diferença? Sim, o Java procura o primeiro catch que pode 





trabalhar a Exception adequada. 








Repare que RuntimeException herda de Exception e, portanto, 
deve vir antes da mesma na ordem de catches. 
Caso ela viesse depois, ela nunca seria invocada, pois o Java verificaria que 











toda RuntimeExceptioné Exceptione Exception teria tratamento de 





preferência (por sua ordem). O exemplo a seguir não compila por este motivo: 


void metodo1() { 

try { 
metodo2(); 

+ catch(IDException ex) { 
System.out.printlin("io''); 

} catch(Exception ex) { 
System.out.println('"exception qualquer"); 

+ catch(RuntimeException ex) { 
// não compila pois jamais será executado 
System. out .println("runtime"); 


Cuidado também com exceptions nos inicializadores: 


class AcessoAoArquivo { 
// não compila, pois ao instanciar, pode dar IOException, 
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// mas o construtor não fala nada 
private InputStream is = new FileInputStream("entrada.txt'); 





Nesses casos, precisamos dizer no construtor que a Exception pode ser 
jogada: 


class AcessohoArquivo 1 
private InputStream is = new FileInputStream("entrada.txt'); 


AcessohoArquivo() throws I0Exceptiont 
// estou avisando os clientes dessa classe 
// que ao instanciar pode dar essa exception 
// e agora compila 


1) Qual classe podemos colocar no código a seguir para que ele compile? 


ı import java.io.*; 


2 class X { 

3 InputStream y() throws NOME AQUI { 

4 return new FileInputStream("a.txt'); 
5 + 

6 void z() throws NOME AQUIT 

7 InputStream is = y(); 

8 is.close(); 

9 } 

1 + 


%* %* %* 








java.io. IOException java. sql.SQLException 








java. lang.Exception* java. lang. IndexOutOfBoundsException 


%* 





java.io.FileNotFoundException 


2) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 
2 void m2() { 
3 System. out.println("e"); 
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4 int[] [lx = new int[15] [20]; 
5 System.out.println("f'); 

6 F} 

E void mO { 

8 System.out.println(''c''); 

9 m2(); 

10 System.out.println("d"); 

11 } 

12 public static void main(String[] args) { 
3 System.out.println("a"); 

14 new AQ .m(); 

15 System.out.println("b"); 

16 } 

y } 


a) Não compila. 


b) Compila e imprime acefdb. 





c) Compila e imprime ace e joga uma Exception. 





d) Compila e imprime acedb e joga uma Exception. 





e) Compila e imprime ace, joga uma Exception e imprime db. 


3) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir: 


ı class A { 

2 void m2() { 

3 System.out.println("e"); 
4 int[] x = new int[15]; 

š x[20] = 13; 

6 System.out.println("f"); 
7 } 

8 void m() { 

9 System.out.printlin('"c'); 
10 m2(); 

u System.out.println("d"); 
12 } 

3 public static void main(String[] args) { 
14 System.out.println("a"); 
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15 new A().m(); 

16 System.out.printin("b''); 
17 } 

18 + 


a) Não compila. 
b) Compila e imprime acefdb. 


c) Compila e imprime ace e joga uma 





Exception. 





d) Compila e imprime acedb e joga uma Exception. 





e) Compila e imprime ace, joga uma Exception e imprime db. 


4) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir, 


sem que o arquivo exista? 


ı class A { 

2 void m2() { 

3 System. out.println("e"); 

4 new java.io.FileInputStream("a.txt"); 
5 System.out.printin("f''); 

6 } 

: void m() { 

8 System.out.printin('c''); 

9 m2(); 

10 System. out.println("d"); 

nu } 

12 public static void main(String[] args) { 
3 System. out.println("a"); 

14 new A(O) .m(); 

15 System. out.println("b"); 

16 } 

7} 


a) Não compila. 
b) Compila e imprime acefdb. 


c) Compila e imprime ace e joga uma 





Exception. 





d) Compila e imprime acedb e joga uma Exception. 
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e) Compila e imprime ace, joga uma Exception e imprime db. 


5) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir, 


sem que o arquivo exista? 


ı class A { 

2 void m2() throws java.io.FileNotFoundException { 
3 System.out.println('e''); 

4 new java.io.FileInputStream("a.txt'); 

5 System.out.printin("f'); 

6 + 

7 void m() throws java.io.I0Exception { 

8 System.out.printlin(''c''); 

9 m2(); 

10 System.out.println("d"); 

11 } 

12 public static void main(String[] args) 

13 throws java. io.FileNotFoundException { 
14 System.out.println("a'); 

15 new AQ .m(); 

16 System.out.printlin("b'); 

17 } 

18 } 


a) Não compila. 


b) Compila e imprime acefdb. 





c) Compila e imprime ace e joga uma Exception. 





d) Compila e imprime acedb e joga uma Exception. 





e) Compila e imprime ace, joga uma Exception e imprime db. 


6) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir, 


sem que o arquivo exista? 


ı class A { 

2 void m2() throws java.io.FileNotFoundException { 
3 System.out.println('e''); 

4 new java.io.FileInputStream("a.txt'); 
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System.out.printin("f''); 
} 


void m() throws java.io.FileNotFoundException { 
System. out.println("c"); 
m2(); 
System.out.printin("d'); 
public static void main(Stringl[] args) throws 
java.io.I0Exception { 
System.out.printin("a''); 
new AQ .m(); 
System.out.printin("b''); 


a) Não compila. 


b) Compila e imprime acefdb. 





c) Compila e imprime ace e joga uma Exception. 





d) Compila e imprime acedb e joga uma Exception. 





e) Compila e imprime ace, joga uma Exception e imprime db. 


7) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir, 


sem que o arquivo exista? 


ı class A { 


2 
3 
4 


5 
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void m2() throws java.io.FileNotFoundException { 
System.out.printin('e''); 
new java.io.FileInputStream("a.txt'); 
System.out.printin("f''); 

} 

void m() throws java.io.FileNotFoundException { 
System. out.println("c"); 
try t 

m2(); 

} catch(java.io.FileNotFoundException ex) { 
} 
System.out.printin("d'); 
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14 } 

15 public static void main(String[] args) throws 
16 java.io.I0Exception { 

17 System.out.println("a"); 

18 new A().m(); 

19 System.out.printlin("b'); 

20 } 

21 } 


a) Não compila. 


b) Compila e imprime acefdb. 





c) Compila e imprime ace e joga uma Exception. 





d) Compila e imprime acedb e joga uma Exception. 


e) Compila e imprime acedb. 





f) Compila e imprime ace, joga uma Exception e imprime db. 
8) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir, 
sem que o arquivo exista? 


ı class MyException extends RuntimeException { 


2 


3 5 

4 class A 1 

5 void m2() throws java.io.FileNotFoundException { 
6 System.out.println("e''); 

7 new MyException(); 

8 System.out.printlin("f'); 

9 } 

10 void m() throws java.io.FileNotFoundException { 
n System.out.printlin("c'); 

12 try É 

13 m2(); 

14 + catch(java.io.FileNotFoundException ex) { 
15 } 

16 System.out.println("d"); 

17 } 

18 public static void main(String[] args) throws 
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java.io.I0Exception 1 
System.out.printin("a''); 
new AO .m(); 
System.out.printin("b''); 


a) Não compila. 


b) Compila e imprime acefdb. 


c) Compila e imprime ace e joga uma 


d) Compila e imprime 








Exception. 





acedb e joga uma Exception. 


e) Compila e imprime ace, joga uma Exception e imprime db. 


9) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir, 


sem que o arquivo exista? 


ı class MyException extends RuntimeException { 


2 
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4 class A 1 
void m2() throws java.io.FileNotFoundException { 


5 


6 


System.out.println('e''); 

throws new MyException(); 

System.out.printin("f''); 
} 


void m() throws java.io.FileNotFoundException { 


System. out.println("c"); 
try { 
m2(); 


} catch(java.io.FileNotFoundException ex) { 


F} 
System.out.printin("d'); 


} 


public static void main(String[] args) throws 


java.io.I0Exception { 
System. out.println("a"); 
new AQ .m(); 
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22 System.out.println("b"); 


a) Não compila. 


b) Compila e imprime acefdb. 





c) Compila e imprime ace e joga uma Exception. 





d) Compila e imprime acedb e joga uma Exception. 





e) Compila e imprime ace, joga uma Exception e imprime db. 
10) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir, 
sem que o arquivo exista? 


ı class MyException extends RuntimeException { 


2 


3 5 

4 class A 1 

5 void m2() throws java.io.FileNotFoundException { 
6 System.out.println('e''); 

7 boolean sim = true; 

8 if(sim) throws new MyException(); 

9 System.out.println("f'); 

10 } 

u void m() throws java.io.FileNotFoundException { 
12 System.out.println("c"); 

3 try É 

14 m2(); 

15 } catch(java.io.FileNotFoundException ex) { 
16 } 

17 System.out.println("d"); 

18 } 

19 public static void main(String[] args) throws 
20 java.io.I0Exception { 

21 System.out.println("a"); 

22 new A().m(); 

23 System.out.println("b"); 

24 } 

as } 
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a) Não compila. 


b) Compila e imprime acefdb. 





c) Compila e imprime ace e joga uma Exception. 





d) Compila e imprime acedb e joga uma Exception. 





e) Compila e imprime ace, joga uma Exception e imprime db. 


11) Escolha a opção adequada ao tentar compilar e rodar o arquivo a seguir, 


sem que o arquivo exista? 


ı class MyException extends RuntimeException { 


2 


3 5 

4 class A 1 

5 void m2() throws java.io.FileNotFoundException { 
6 System.out.printin('e''); 

7 boolean sim = true; 

8 if(sim) throw new MyException(); 

9 System.out.printin("f''); 

10 } 

u void m() throws java.io.FileNotFoundException { 
12 System. out.println("c"); 

13 try 1 

4 m2(); 

15 } catch(java.io.FileNotFoundException ex) { 
16 E; 

17 System.out.printin("d'); 

18 } 

19 public static void main(String[] args) throws 
20 java.io.I0Exception { 

a System.out.printin("a''); 

22 new AO .m(); 

23 System. out.println("b"); 

24 } 

2 } 


a) Não compila. 


b) Compila e imprime acefdb. 
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c) Compila e imprime ace e joga uma Exception. 





d) Compila e imprime acedb e joga uma Exception. 





e) Compila e imprime ace, joga uma Exception e imprime db. 


10.5 RECONHECA CLASSES DE EXCEÇÕES COMUNS E 
SUAS CATEGORIAS 
Para a prova, é necessário conhecer algumas exceptions clássicas do Java. Na 


sequência, vamos conhecer essas exceptions e entender em que situações elas 


ocorrem. 


A Exception e IndexOutOfBoundsExcep- 
ion 





Um ArrayIndexOutOfBoundsException ocorre quando se tenta 
acessar uma posição que não existe em um array. 


class Teste { 
public static void main(Stringl[] args) { 
int [] array = new int[10]; 
array [10] = 10; // Aqui ocorre 
// Array IndexOut0fBoundsException. 


Da mesma maneira, quando tentamos acessar uma posição 
não existente em uma lista , a exception é diferente, no caso 





IndexOutOfBoundsException 


class Teste { 
public static void main(Stringl[] args) { 
ArrayList<String> lista = new ArrayList<String>(0; 


//Aqui ocorre Index0Out0fBoundsException 
String valor = lista.get(2); 
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NullPointerException 


Toda vez que o operador . é utilizado em uma referência nula, um 








NullPointerException é lançado 


class Teste 1 
public static void main(Stringl] args) 1 
String s = null; 
s.length(); // Aqui ocorre uma NullPointerException 


ClassCastException 


Quando é feito um casting em uma referência para um tipo incom- 
patível com o objeto que está na memória em tempo de execução, ocorre um 





ClassCastException 


class Teste 1 
public static void main(Stringl] args) 1 
Object o = "SCJP"; // String 
Integer i = (Integer)o; // Aqui ocorre 
// ClassCastException. 


NumberFormatException 


Um problema comum que o programador enfrenta no dia a dia é ter que 
“transformar” texto em números. A API do Java oferece diversos métodos 
para tal tarefa. Porém, em alguns casos não é possível “parsear” o texto, pois 
ele pode conter caracteres incorretos. 


class Teste 1 
public static void main(Stringl] args) 1 
String s = "ABCD1"; 


// Aqui ocorre um NumberFormatException. 
int i = Integer.parseInt(s); 
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IlegalArgumentException 


Qualquer método deve verificar se os valores passados nos seus parâmet- 
ros são válidos. Se um método constata que os parâmetros estão in- 
válidos, ele deve informar quem o invocou que há problemas nos val- 
ores passados na invocação. Para isso, é aconselhado que o método lance 





IllegalArgumentException. 


class Teste { 
public static void main(Stringl[] args) 1 
try É 
divideEImprime(5,0); 
} catch (TllegalArgumentException e) { 
// tratamento. 


public static void divideEImprime(int i, int j) 1 
if(j == 0) { // Evita dividir por zero. 
throw new IllegalArgumentException(); 


} 
System. out.println(i/j); 


IllegalStateException 


Suponha que uma pessoa possa fazer três coisas: dormir, acordar e andar. 
Para andar, a pessoa precisa estar acordada. A classe Pessoa modela o com- 
portamento de uma pessoa. Ela contém um atributo boolean que indica se 
a pessoa está acordada ou dormindo e um método para cada coisa que uma 
pessoa faz ( dormir (), acordar () e andar ()). 

O método andar() não pode ser invocado enquanto a pessoa está 
dormindo. Mas, se for, ele deve lançar um erro de execução. A bib- 
lioteca do Java já tem uma classe pronta para essa situação, a classe é a 
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IllegalStateException. Ela significa que o estado atual do objeto não 
permite que o método seja executado. 


class Pessoa { 
boolean dormindo = false; 


void dormir() { 
this.dormindo = true; 
System.out.printin("dormindo...'); 

} 

void acordar() { 
this.dormindo = false; 
System.out.println("acordando..."); 

f 

void andar() { 
if (this.dormindo) { // Só pode andar acordado. 

throw new IllegalStateException("Deveria estar 
acordado!''); 


System.out.println("andando...''); 


ExceptionInInitializerError 


No momento em que a máquina virtual é disparada, ela não carrega todo 
o conteúdo do classpath, em outras palavras, ela não carrega em memória 
todas as classes referenciadas pela sua aplicação. 

Uma classe é carregada no momento da sua primeira utilização. Isso se dá 
quando algum método estático ou atributo estático são acessados ou quando 
um objeto é criado a partir da classe em questão. 

No carregamento de uma classe, a JVM pode executar um trecho de 
código definido pelo programador. Esse trecho deve ficar no que é chamado 
bloco estático. 


class A { 
static { 
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// trecho a ser executado no carregamento da classe. 


E totalmente possível que algum erro de execução seja gerado no bloco 
estático. Se isso acontecer, a JVM vai “embrulhar” esse erro em um 








ExceptionInInitializerError e dispará-lo. 
Esse erro pode ser gerado também na inicialização de um atributo estático 
se algum problema ocorrer. Exemplo: 


class A 1 
static { 
if (true) 
throw new RuntimeException("nao vou deixar nao..."); 
E; 
} 


public class Teste { 


public static void main(String[] args) { 
new A(); 


Gera o erro de inicialização: 


Exception in thread "main" java.lang.ExceptionInInitializerError 
at Teste.main(Teste.java:11) 

Caused by: java.lang.RuntimeException: nao vou deixar nao... 
at A.<clinit>(Teste.java:4) 


1 more 
class P{ 
static int a = Integer .parseInt ("a"); 
} 
StackOverflowError 


Todos os métodos invocados pelo programa Java são empilhados na Pilha 
de Execução. Essa pilha tem um limite, ou seja, ela pode estourar: 
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class Teste { 
public static void main(Stringl] args) 1 
metodoSemFim(); 


static void metodoSemFim() ( 
metodoSemFim() ; 


Repare que, nesse exemplo, o metodoSemF im () chama ele mesmo (re- 
cursão). Do jeito que está, os métodos serão empilhados eternamente e a pilha 


de execução vai estourar. 


NoClassDef FoundError 


Na etapa de compilação, todas as classes referenciadas no código-fonte 
precisam estar no classpath. Na etapa de execução também. O que será que 
acontece se uma classe está no classpath na compilação mas não está na exe- 





cução? Quando isso acontecer será gerado um NoClassDefFoundError. 


Para gerá-lo, podemos criar um arquivo com duas classes onde uma ref- 
erencia a outra: 


class OutraClasse 1 


} 
class Teste { 
public static void main(String[] args) { 
new OutraClasse(); 


Compilamos o arquivo, gerando dois arquivos .class. Aí apagamos o 


arquivo OutraClasse.class. Pronto, o Java não será capaz de encontrar 





a classe, dando um erro, NoClassDefFoundError. 
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OutOfMemoryError 


Durante a execução de nosso código, o Java vai gerenciando e limpando 
a memória usada por nosso programa automaticamente, usando o garbage 
collector (GC). O GC vai remover da memória todas as referências de objetos 
que não são mais utilizados, liberando o espaço para novos objetos. Mas o que 
acontece quando criamos muito objetos, e não os liberamos? Nesse cenário, 
o GC não vai conseguir liberar memória, e eventualmente a memória livre irá 





acabar, ocasionando um OutOfMemoryError. 

O código para fazer um erro do gênero é simples, basta instanciar infinitos 
objetos, sem permitir que o garbage collector jogue-os fora. Fazemos isso com 
Strings para que o erro aconteça logo: 


void metodo() 1 
ArrayList<String> objetos = new ArrayList<String>0); 


String atual = ""; 
while(true) { 
atual += " ficou maior"; 


objetos.add (atual); 


1) Escolha a opção adequada que indica o Throwable que ocorrerá no 
código a seguir: 


ı class A { 

2 public static void main(Stringl] args) 1 
3 main(args); 

4 } 

5 } 





a) IndexOutOfBoundsException 





b) ArrayIndexOutOfBoundsException 


c) NullPointerException 








d) OutOfMemoryError 





e) StackOverflowError 
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f) ExceptionInInitializationError 








2) Escolha a opção adequada que indica o Throwable que ocorrerá no 
código a seguir: 


ı import java.util.x*; 


2 class A { 

3 public static void main(Stringl] args) 1 

4 ArrayList<String> strings = new ArrayList<String>0); 
5 for (int i=0;i<10;i++) 

6 for(int j=0;j<10; i++) 

7 strings.add("string "+ i +" " + j); 

8 System.out.printin(strings.get (99999)); 

9 E; 

10 + 


a) IndexOutOfBoundsException 








b) ArrayIndex0OutOfBoundsException 





c) NullPointerException 








d) OutOfMemoryError 





e) StackOverflowError 





f) ExceptionInInitializationError 
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CAPÍTULO 11 


Boa prova 


Agora que você acabou todo o livro, está na hora de revisar os pontos em 
que ficou com dúvida. Refaça os exercícios para reforçar o conteúdo e faça 
diversos simulados como os citados no começo do livro. 

Para efetuar a prova, primeiro compre o voucher no site da Oracle. 
Busque no site dela pelo código da prova que deseja fazer, a Java SE 7 Pro- 
grammer I (1Z0-803), adicione a mesma ao carrinho e efetue a compra. 

Hoje o pagamento é feito via boleto enviado por email, tome cuidado pois 
ele pode cair em sua caixa de spam, e tome ainda mais cuidado para não pagar 
um boleto de spam. Depois de confirmado o recebimento por parte da Oracle 
você receberá um código de confirmação com o qual será capaz de agendar a 
data e local de sua prova. 

Lembre-se de reler os pontos que tem mais dificuldade no dia que ante- 
cede a prova. 
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Mais uma vez, boa prova. Desejo que todo esse processo tenha aberto sua 


mente sobre como a linguagem Java funciona, seus limites e suas caracterís- 
ticas gerais. 


Depois dessa certificação, que venha a próxima, que exige ainda mais con- 
hecimento da linguagem e de APIs fundamentais da mesma. 


366 


CAPÍTULO 12 


Respostas dos Exercícios 


3.1 - Exercícios 


1. A resposta certa é (c). A variável i declarada no for só é visível dentro 
do for. 


2. A resposta certa é (c), a variável x declarada como parâmetro do 
método main efetua um shadowing. Nesse instante, ao dizermos x 
= 200, tentamos atribuir um int a um array de String, erro de 
compilação. 


3. À resposta certa é (d). Isso porque o acesso à variável estática pode 
ser feito através da instância da classe ou diretamente caso seja uma 
variável estática sendo acessada por um método estático. 
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3.2 - Exercícios 


1. À resposta certa é (a). O arquivo não compila pois não podemos ter 


um import após a definição de uma classe. 


A resposta certa é (e). O código compila e roda, não imprimindo nada, 
pois não chamamos o método Teste. 


. A resposta certa é (c). O código compila sem erros: a ordem package 


e import está adequada, e os tipos são opcionais dentro de um arquivo 
java. 


A resposta certa é (d). Uma vez que o arquivo chama A. java, o único 
tipo público que pode existir dentro dele deve se chamar A, o que não 
é verdade: tentamos definir um tipo B público. 


3.3 - Exercícios 
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. O método main não pode devolver int nem Void. Ele também 


deve ser público e só receber um argumento: um array (ou varargs) de 
String. Portanto: 





public static void main(String... args) 
A resposta certa é (a). 
Para compilar, estamos trabalhando com arquivos e diretórios, por- 


tanto javac b/A. java; enquanto, para rodar, estamos pensando em 


pacotes e classes: java b.A. A resposta certa é (d). 


. Ao rodar sem argumentos, nosso array tem tamanho zero, por- 


tanto, ao tentar acessar seu primeiro elemento recebemos um 
NullPointerException na linha 5. A resposta certa é (c). 








Para rodar um programa dentro de um jar sem ter um manifesto, 
devemos usar o classpath customizado. Colocamos o jar no classpath 
e dizemos qual classe desejamos rodar: java -cp programa. jar 
b.A. À resposta certa é (f). 
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5. Durante a compilação, para adicionar o arquivo programa. jar ao 
classpath, devemos usar -cp programa. jar e, para especificar o ar- 
quivo adequado, usamos b/A. java. A opção que apresenta essas duas 
características é javac -cp programa.jar:. b/A. java. A re- 
sposta certa é (h). 


3.4 - Exercícios 


1. A resposta certa é (a). Ocorre um erro de compilação na classe Teste 
ao tentar importar uma classe não acessível a outros pacotes. 


2. À resposta certa é (a). O erro de ambiguidade é dado no import e não 
na utilização, portanto o arquivo não compila. 


3. A resposta certa é (c). Não existe ambiguidade uma vez que o import 
específico tem preferência em cima do x 


4. À resposta certa é (c). Não há erro de ambiguidade, simplesmente um 
import é desnecessário e não gera erro nenhum, apenas um warning. 


5. A resposta certa é (e). O arquivo B compila pois é uma classe normal. O 
arquivo C não compila pois tenta acessar B, que está em outro pacote, 
mas lembre-se que devemos acessar os pacotes diretamente, não existe 
subpacote. O mesmo vale para A. Portanto, nem A nem C compilam. 


6. A resposta certa é (d). Um pacote pode ter nome começando com 
maiúsculo, isso não afeta em nada. Mas não é o padrão. O nome de 
variáveis locais e parâmetros não afetam a assinatura de um método 
em Java. Uma classe não precisa ser pública para ser rodada. Portanto 
o código compila e roda. 


7. A resposta certa é (b). Não importamos a classe A, somente seus mem- 
bros, erro de compilação ao tentar referenciá-la na linha 5. 


8. A resposta certa é (a). import static é o uso adequado, e não 


static import, erro na linha 3. 
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9. A resposta certa é (a). B não compila pois tenta acessar uma classe 


do pacote padrão (sem nome). Classes do pacote padrão só podem ser 
acessadas por outros tipos do pacote padrão. Não compila. 


4.1 - Exercícios 
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1. A resposta certa é (d). O código compila e imprime 100. Podemos ter 


espaços em branco desde que não quebre uma palavra-chave, nome de 
método, classe etc. ao meio. Onde pode ter um espaço em branco, pode 
haver vários. 


. À resposta certa é (a). O código não compila pois tentamos acessar 


a variável idade que pode não ter sido inicializada. Não é certeza 
(somente se cair no if ela será inicializada). 


. A resposta certa é (f). Não compila, do lado direito da atribuição temos 


um array de boolean e do lado esquerdo uma variável simples do tipo 
boolean. 


A resposta certa é (b). Imprime false pois um array de tipos primi- 
tivos após a inicialização tem seus valores com o valor padrão do tipo. 
Para numéricos é 0, para booleané falsee para referências é null. 


. A resposta certa é (b). Não compila pois boolean em Java só pode ser 


false ou true. 


. A resposta certa é (a). O número octal 09 não existe. Você não precisa 


aprender a transformar uma base em outra, mas é importante lembrar 
que binários são compostos de os e ıs, octais são compostos de os até 
7s, hexadecimais são de os até gs e As até Fs (maiúsculo ou minúsculo). 
O caractere _ é permitido desde que dos dois lados dele tenhamos al- 
garismos válidos, que é o caso de 1 000. Portanto, o único número 
inválido é o9 (por curiosidade, o número 9 em base octal é 011). 


. A resposta certa é (c). Compila e imprime o alfabeto pois caracteres 


são números em Java. 
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8. A resposta certa é (e). instanceof não é palavra reservada: note a 
letra maiúscula no meio dela. Nenhuma palavra-chave em Java possui 
caractere maiúsculo. 


9. A resposta certa é (d). Compila e roda, não imprimindo nada. Lembre- 
se que os identificadores são case-sensitive. 


4.2 - Exercícios 


1. À resposta certa é (e). Imprime 47, pois a atribuição é por cópia do 
valor. 


2. À resposta certa é (f). Imprime 48, pois a atribuição de objetos é feita 
por cópia da referência, criamos somente um único objeto do tipo B. 


4.3 - Exercícios 


1. A resposta certa é (d). Não existe conflito de nomes entre variável mem- 
bro e método ou variável membro e variávellocal. Ao invocar o método 
c, por causa do shadowing da variável c, não acessamos a variável 
membro, sem alterá-la. O resultado é a impressão dos valores 10 e 10 
novamente. 


4.4 - Exercícios 


1. À resposta certa é (c). Compila e não podemos falar nada. 


2. A resposta certa é (b). Somente 10 objetos podem ser garbage coletados 
pois o último continua referenciado pela variável b. 


3. A resposta certa é (b). O código compila, mas como não chamamos 
nenhum construtor, o único objeto criado que se assemelha a B, porém 
não é B, é um array do tipo B, com 100 espaços. Nenhum objeto é 
criado. Note que para criar devemos, por padrão, invocar o construtor. 
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4.5 - Exercícios 


1. À resposta certa é (b). O código compila e devido à regra de sempre 


invocar o mais específico, ele sempre invoca o método sem argumentos. 
Portanto, o resultado é vazio/vazio. Lembre-se que em Java a ordem de 
definição de métodos não importa para a invocação. Já a ordem das 
variáveis pode importar, caso uma dependa da outra. 


. A resposta certa é (c). O código compila e imprime 2. 
. A resposta certa é (e). O código compila e imprime 2. 


. A resposta certa é (e). O código compila e imprime 2. Esse é o caso 


absurdo onde o array é tanto um Object quanto um array de Object. 
Por padrão o Java tratará como um array de Object. 


4.6 - Exercícios 


1. À resposta certa é (e). O array começa na posição o, portanto, o 


primeiro caractere removido se encontra na posição 2,0 i. Ele remove 
todos os caracteres até a posição 3, exceto o da posição 3, portanto so- 
mente o i é removido. 


. A resposta certa é (a). Os dois métodos retornam -1 quando não en- 


contram nada, portanto, o segundo resultado é -2. Como a posição 
começa em 0, o resultado das letras e são 5e 8, totalizando 13. 


4.7 - Exercícios 
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1. A resposta certa é (a). Não compila, pois length () é um método de 


String, diferente dos arrays em que length é um atributo. 





2. A resposta certa é (b). Dá NullPointerException! msgé null e 





não dá pra chamar isEmpty em null. 





3. A resposta certa é (a). Não compila pois a variável não foi inicializada. 
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10. 


11. 


12 


13. 


14. 


A resposta certa é (a). ‘Caelum‘ e ‘Caelum - Ensino e Inovação: 


. A resposta certa é (e). Compila e imprime Bem-vindo null 


. A resposta certa é (b). Não compila pois a String não foi inicializada 


A resposta certa é (b). Não compila por outro motivo: a variável vazio 
não é estática. 


. A resposta certa é (e). Compila e imprime Bem-vindo null. 





. A resposta certa é (d). Dá NullPointerException ao tentar criar a 





segunda String. 
A resposta certa é (c). O código compila e imprime uda. 


A resposta certa é (a). Não compila pois String possui diversos con- 
strutores que recebem um argumento: o compilador não sabe qual de- 
les você deseja invocar pois os tipos que são argumentos do constru- 
tor não possuem herança entre si (um não herda necessariamente do 
outro). 


A resposta certa é (c). Nenhuma das alternativas dadas com número, 
pois primeiro ele soma valor e dividePor, imprimindo 14. A conta 
de divisão é feita entre int, devolvendo um int de valor 2. Quando 
esse número é atribuído a um double, continua sendo 2. Portanto, 
imprime 14e 2.0. 


A resposta certa é (a). O código não compila pois o método replace 
possui duas maneiras de ser invocado: com dois chars ou com duas 
Strings. Foram passados uma String eum char, método que não 
existe. 


A resposta certa é (a). Pensamos que pode ser gualherme, mas lem- 
bramos que String em Java é imutável e ela não foi reatribuída. O 
= dá uma impressão de reatribuição de parte da String, mas isso 
não existe em Java, o lado esquerdo de uma atribuição deve ser sempre 
uma variável e não uma chamada a um método. Por isso, a linha do 
substring não compila. 
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5.1 - Exercícios 
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. As opções (a) e (c) não compilam e precisam do casting mesmo com 


short e char tendo 2 bytes. 


Na opção (a), pode ocorrer deo short ser negativo e, portanto, não 
caber no intervalo dos chars. Na opção (c), o char pode ser muito 
grande e sair fora do alcance dos positivos do short. 


A opção (b) compila, poiso char que possui 2 bytes pode ser atribuído 
paraum long que possui 8 bytes. 


. A resposta certa é (b). Análise linha a linha: 


- divisão inteira: ii vale 1 


- divisão inteira, que depois é promovido a double: iz vale 1.0 


divisão double: i3 vale 1.5 


x vale oL e d vale o.o (duas promoções) 


O resultado é 3.5 


. À resposta certa é (d). Mesmo c sendo null, por estarmos us- 


ando o operador «,a segunda parte da expressão (c.getPreco() > 





10000) será avaliada, causando uma NullPointerException na 





chamada do método getPreco() caso cseja null. Poderíamos 
evitar isso usando o operador de curto-circuito, ss. 


. A resposta certa é (d): i e c. 


. A resposta certa é (d). Pode-se utilizar o operador booleano de ou ex- 


clusivo ^: 
if (trem ^ carro) { 
ARTIT 
} 
.1° linha: ArithmeticException: / by zero 2° linha: 








Infinity 3°linha: Infinity 4° linha: -Infinity 
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7. A resposta certa é (b). O código não compila na linha 7. O compi- 
lador não tem certeza se a variável y vai ser iniciada sempre. Como a 
declaração é feita e o único valor atribuído é dentro do for, o compi- 
lador não tem certeza seo for vai ser executado mesmo. 


8. A resposta certa é (d). 


9. A resposta certa é (e). Compila, roda e imprime A75. Cuidado ao com- 
pilar e rodar pois alguns caracteres podem precisar ser escapados pelo 
seu shell ou bash (não cobrado na prova). 


5.1 - Exercícios 


1. Não compila! Toda conta devolve no mínimo um int. O resultado de 
b1 + b2 é int. Podemos fazer casting ou declarar b3 como int. 


5.1 - Exercícios 


1. A resposta certa é (c). Não compila, pois toda conta devolve no mínimo 
um int, eum int não cabe em um byte. 


5.1 - Exercícios 


1. A resposta certa é (c), 3. 


O for externo vai contar de o a 5, mas dentro do for tem um if que 
pré-incrementa o i. Esse if vai quebrar o loop no momento que o 
valor retornado for divisível por 3, isto é, quando i valer 3 nesse caso. 


2. A resposta certa é (a), 1. 


Dessa vez, o valor de i será usado no if e só depois incrementado. 
Como o resto de o dividido por qualquer número também é 0,0 for 
só executa uma vez. Mas o valor de i ainda será incrementado, impri- 


mindo o valor de 1. 
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10. 


Vai imprimir 3. 
O for externo vai contar de o a 5, mas dentro do for tem um if que 


pré-incrementa o i. Esse if vai quebrar o loop no momento em que o 
valor retornado for divisível por 3, isto é, quando i valer 3 nesse caso. 


. A resposta certa é (f). A segunda linha do método main não compila 


pois estoura o limite de byte. 


. Aresposta certa é (1). O código não compila pois não podemos declarar 


um char negativo. 


A resposta certa é (f). O código compila e imprime um outro valor (65 
+3). 


. A resposta certa é (b). O código compila e joga uma exception por 


causa da divisão inteira (são ints) por zero. 


A resposta certa é (c). O código compila e imprime positivo infinito. A 
precedência de operadores é primeiro a divisão, por isso compila. 


A resposta certa é (a). Não compila, não há comparação entre 
boolean e números. 


5.2 - Exercícios 


1. 
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A resposta certa é (a). Não compila pois o resultado do parenteses é 
uma String que não possui o operador de divisão. 


A resposta certa é (b). true==falseé false. O inverso disso é 
true. Concatenando com "" o resultado é true, que é iguala true. 
Portanto, o operador ternário devolve 1 que é diferente de o, imprim- 
indo false. 
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5.3 - Exercícios 


1. A resposta certa é (b). O código compila e imprime true e false. 


2. A resposta certa é (d). Compila e imprime false, false, uma vez 
que a String 2 vale s. 


3. A resposta certa é (c). Compila e imprime true, true. Por mais que 
substring devolva uma nova String, nesse caso ele devolveu a String 
inteira, a própria String. 


4. A resposta certa é (c). O código imprime true e true. Note que o 
método equals não foi sobrescrito. 


5. A resposta certa é (a). O código não compila pois D não é do tipo C. 


5.4 - Exercícios 


1. À resposta certa é (c). O código compila normalmente e imprime 0 
caso não seja passado nenhum argumento. 


2. À resposta certa é (a). O código não compila pois a variável valor é 
final e não pode ser alterada, mas tentamos efetuar uma atribuição 
dentro do if. 


3. A resposta certa é (a). O código não compila pois tenta atribuir 15 a 
uma variável e conferir o valor 15 como se fosse um boolean. 


4. A resposta certa é (a). O código não compila pois não existe palavra- 
chave elseif. Devemos fazer um else if para compilar. 


5. A resposta certa é (b). O código não compila pois o else não está 
aplicado ao if: para ser aplicado ao if, ele deve vir imediatamente 
após seu bloco. Como o if não possui chaves, somente a primeira 
linha pertence a ele. 
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5.5 - Exercícios 


1. À resposta certa é (d). Ao rodar com 5 argumentos, o código imprime 


mais argumentos. 





. A resposta certa é (a). O código não compila pois tamanhoEsperado 


não é uma constante. Somente podemos verificar case de switch 
em variáveis finais inicializadas diretamente. 


. A resposta certa é (b). A String "42" é uma String uma vez que 


ela está entre aspas. Portanto, o código imprime Guilherme. 


. A resposta certa é (a). A sintaxe do case é com : enão com {, 0 


código não compila. 


. À resposta certa é (a). O código não compila pois o case não aceita 


expressões como < x, mas sim um valor definido em tempo de com- 
pilação. 


. À resposta certa é (a). O código não compila pois há código que não 


será executado após break. 


6.1 - Exercícios 
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1. A resposta certa é (f). Não faz sentido ter colchetes antes da declaração 


do tipo, portanto []int x não compila. 


. A resposta certa é (b). A segunda linha não compila pois, ou você passa 


o tamanho, ou passa os valores. 


. À resposta certa é (x). O programa não compila pois a segunda e a 


terceira linha tentam redefinir uma variável já definida. Caso o nome 
da variável seja corrigido, o código compila e imprime nada ao rodar 
(um array pode ter tamanho zero). 


. A resposta certa é (c). O programa inicializa i para o tamanho do ar- 


ray, acessando uma posição inexistente. Portanto dá erro em execução 
(exception). 
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5. A resposta certa é (b). Não tenha medo de simular o código na mão. 





Simule a memória e perceba que dá uma Exception. 


6. A resposta certa é (f). Não tenha medo de simular o código na mão. 
Simule a memória e perceba que o resultado é 2, -5. Para isso, de- 
senhe os três espaços do array, aponte os valores iniciais o e continue 
atribuindo valores, executando o código. 


Durante a prova, simular os arrays e os ponteiros é ideal para não se 
perder em códigos complexos de referências e valores com arrays. 


7. A resposta certa é (d). Compila e imprime true: note que não existe 
criação de um novo array, nós simplesmente temos duas referências ( 


valores e vals para o mesmo array na memória). 


8. As respostas corretas são (a), (b), (f) e (j). 


6.2 - Exercícios 


1. A resposta certa é (a). Não compila pois, ao inicializarmos o array zyx, 
utilizamos um array de uma única dimensão. 


2. A resposta certa é (f). Na posição 2, temos o array z, que tem 30 casas, 
portanto temos o resultado 30. 


3. A resposta certa é (g). O código compila e imprime 30 normalmente. 
Não há problema algum em apontar para um novo array. 


4. A resposta certa é (a). Nesse exemplo, é guardado um valor double 
em uma das posições do array de int idades. Isso está incorreto 
portanto não compila. 


Na declaração do array de duas dimensões tabela, são informa- 
dos os tamanhos das dimensões. Errado pois os tamanhos devem ser 
definidos na inicialização e não na declaração. 


Na inicialização do array cubo, não foi colocado o tamanho de nen- 
huma das dimensões. 
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6.3 - Exercícios 


1. À resposta certa é (a). O código não compila, pois a classe ArrayList 
não foi importada. 


2. A resposta certa é (d). O código roda e imprime true, pois foi re- 


movido um elemento da lista. 


3. À resposta certa é (f). O código roda e imprime 1 pois ele remove o 
primeiro elemento igual ao elemento passado. 


4. A resposta certa é (h). O código compila e imprime 5. 


5. A resposta certa é (a). O código não compila, pois o método toArray 
sem argumentos retorna um array de Object. 


6. A resposta certa é (b). O código inclui os elementos sempre no final da 
ArrayList, portanto imprime a e depois d. 


7. A resposta certa é (a). O código não compila pois a ordem dos 
parâmetros para o método add é int, String. 


8. A resposta certa é (b). O código compila e imprime somente a. Isso 
porque ele executa um next durante o passo de iteração do laço for, 
o que acaba consumindo o segundo elemento sem imprimi-lo. 


9. A resposta certa é (c). O código compila e imprime a, b, c, d, pois o 
laço está alterando o valor referenciado pela variável s, e não o valor 
contido dentro da nossa ArrayList. 


7.1 - Exercícios 


1. À resposta certa é (e). O código compila e ao rodar, a não é maior que 
100, portanto imprime 10. 


2. A resposta certa é (c). O código já compila pois a variável não é final, e 
entra em loop infinito. 
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7.2 - Exercícios 


1. A resposta certa é (a). O código não compila, pois o laço nunca é que- 
brado e nunca chega a executar o código que imprime b. 


2. À resposta certa é (a). O código não compila, pois o código dentro de 


for nunca será executado. 


3. A resposta certa é (a). O código não compila, pois tentamos definir o 
tipo de duas variáveis no nosso for. 


4. A resposta certa é (b). Compila e imprime os valores o até 9. 


5. A resposta certa é (a), O código não compila pois a condição de um 
for deve ser única. 


6. A resposta certa é (e). O código compila e imprime 0 1 1 2 


7.3 - Exercícios 


1. A resposta certa é (b). O código compila e imprime false, pois ele 
sempre entra no laço pelo menos uma vez. 


2. À resposta certa é (b). Compila e entra em loop infinito caso seja pas- 
sado zero, um ou dois argumentos. Não imprime nada caso 3 a 9 argu- 
mentos. Imprime “Finalizou” caso 10 ou mais argumentos. 


3. A resposta certa é (c). O código compila e imprime “o', já que a condição 


é 'i maior que 10. 


4. A resposta certa é (a). O código não compila, pois faltou um ponto e 
vírgula. 


5. A resposta certa é (c). Compila e sai. 
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7.4 - Exercícios 


1. À resposta certa é (a). Quando iteramos por duas coleções ao mesmo 


tempo, podemos usar tanto o for quanto o while, mas o for é 
mais simples, pois passa por todos os elementos já com a inicialização 
e incremento bem definidos dentro do laço. 


. A resposta certa é (a). Usamos o for tradicional (com ou sem 


Iterator) para remover elementos. Poderíamos usar o while, mas 
ele não está na lista de respostas. 


. À resposta certa é (b). Devemos usar o do...while, que garante a 


execução pelo menos uma vez do código. 


. A resposta certa é (d). for, while ou do...while resolvem o 


problema, mas o mais comum éo while. 


. A resposta certa é (b). Não é possível inicializar os valores de um array 


com o enhanced for, portanto usamos o for. 


7.5 - Exercícios 


1. O código não compila pois o segundo if está fora do bloco do for e 


tenta acessar uma variável definida dentro dele. Lembre-se que o es- 
copo de um bloco for sem chaves é uma única instrução, no caso o 
primeiro if else if. 


. A resposta certa é (d). Compila e imprime o até 19, 21 até 24, 26 até 29. 


. A resposta certa é (j). Compila e ao rodar com o argumentos imprime 


1 até 15, fim. 


8.1 - Exercícios 
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1. À resposta certa é (a). O código não compila pois existe um return 


sem valor. 
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2. A resposta certa é (a). Não compila, pois não basta a variável ser final; 
para funcionar no switch, ela tem que ser definida com valor constante 
(sempre 5, por exemplo). 


3. A resposta certa é (b). O código não compila porque o método c re- 
torna um long, e esse long é utilizado como retorno no método a e 
no método b. Ambos precisam de um retorno do tipo int, que não 
tem conversão automática. 


4. A resposta certa é (a). O código não compila, pois não existe retorno 
de método com dois valores como int, int. 


8.2 - Exercícios 


1. A resposta certa é (a). O código não compila pois os métodos não 
possuem tipo de retorno definido. Típica pegadinha: parece focar em 
static mas está focado em outra coisa. 


2. A resposta certa é (b). Imprime x, depois y. 
3. A resposta certa é (d). Compila e imprime z. 


4. À resposta certa é (a). O código não compila pois tenta acessar this 
dentro de um contexto estático. 


5. A resposta certa é (b). O código compila e imprime xe y. 


8.3 - Exercícios 


1. À resposta certa é (a). O código não compila, pois não há sobrecarga 
de método ao alterar só o retorno. 


2. A resposta certa é (c). Compila e imprime 15, 15e 15.0. 
3. A resposta certa é (a). O código não compila. 
4. A resposta certa é (a). Não compila: os métodos não são estáticos. 
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5. A resposta certa é (d). Compila e imprime 3. 
6. A resposta certa é (b). Compila e imprime. 


7. A resposta certa é (a). Não compila, pois tem várias variáveis locais 


(parâmetros) com o mesmo nome. 


8. A resposta certa é (b). Compila e imprime. 


8.4 - Exercícios 


1. A resposta certa é (d). O código compila e joga um 





NullPointerException. 





8.5 - Exercícios 


1. A resposta certa é (a). O código não compila por causa do loop, quando 
um construtor de um tipo chama outro construtor do mesmo tipo em 


loop direto. 


2. A resposta certa é (d). O código não compila pois as classes definem 
parênteses a mais. Cuidado. 


3. A resposta certa é (f). O código compila e não imprime nada. 
4. A resposta certa é (f). O código compila e não imprime nada. 


5. A resposta certa é (e). O código compila e joga exception ao entrar em 
loop infinito. 


8.6 - Exercícios 


1. À resposta certa é (a). Ocorre um erro de compilação na classe Teste 
ao tentar chamar o construtor com acesso default de outro pacote. 


2. A resposta certa é (c). Compila e imprime 3. 
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3. A resposta certa é (b). Não compila na declaração do método private 
public. 


4. A resposta certa é (a). O código não compila, pois a classe A é a própria 
classe do método main, e ela não tem método a. 


5. A resposta certa é (b). Imprime 1, pois o método que recebe String 


não está visível no pacote principal. 

6. A resposta certa é (a). Não compila, pois a palavra default não pode 
ser usada como modificadora de visibilidade de método (ela é usada a 
partir do Java 8 de outra maneira, que não é cobrada nesta prova). 

7. A resposta certa é (b). O código compila e imprime 1 

8.7 - Exercícios 
1. A resposta certa é (b). Compila e imprime o, são duas instâncias de B!. 


2. A resposta certa é (c). Compila e imprime 5. 


3. A resposta certa é (b). Compila e imprime o, existe shadowing aqui no 


setter, 
4. A resposta certa é (c). Compila e imprime 5. 


5. A resposta certa é (d). Compila e imprime 10, existe shadowing aqui no 
setter, então não há problema de a variável ser final. 


8.8 - Exercícios 


1. À resposta certa é (a). Não compila pois somente variáveis podem ter 
aplicadas auto increment e decremento. 


2. A resposta certa é (c). Compila e imprime 151. 
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9.1 - Exercícios 


. Aqui não ocorre sobrescrita. Como os parâmetros são diferentes, 


ocorre uma sobrecarga (não confundir na prova overload com over- 
write). Ou seja, é um overload com herança. 





A reescrita é válida, pois FileNotFoundException é subclasse de 





IOException. 


. A resposta certa é (a). O código não compila pois há um ciclo na her- 


ança. 


A resposta certa é (a). O código não compila pois usa herança múltipla 
de classes, que não existe em Java. 


. A resposta certa é (c). O código compila e não imprime nada. 


A resposta certa é (a). O código não compila pois não existe construtor 
de B ao qual A tenha acesso para herdar do mesmo. 


. A resposta certa é (a). O código não compila pois A não tem acesso a 


variável de B. 


. A resposta certa é (b). Compila e imprime “ 


9.2 - Exercícios 
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1. À resposta certa é (a). O código não compila pois faltou o import de 


java.io.+x. 


. A resposta certa é (b). O código compila e imprime 'b. 


. A resposta certa é (c). O código compila e imprime 'cí 


A resposta certa é (a). O código não compila: c não possui método 
que receba double. 
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5. A resposta certa é (a). Não compila, pois interface não pode ter método 
com corpo da maneira como foi definido aqui. 


6. A resposta certa é (d). Compila e imprime 'd 

7. A resposta certa é (g). Compila e entra em loop. 

8. A resposta certa é (b). Compila e imprime b. 

9. A resposta certa é (a). Não compila, não existe super . x na classe B. 


9.3 - Exercícios 


1. A resposta certa é (a). Não compila pois há um erro de copy e paste nos 
nomes das variáveis. 


2. A resposta certa é (c). Compila e imprime 2. 
3. A resposta certa é (c). Compila e imprime 2. 
4. A resposta certa é (c). Compila e imprime 2. 
5. A resposta certa é (b). Compila e imprime. 


6. A resposta certa é (a). O código não compila pois o método fecha 
não é público. 

7. A resposta certa é (d). O código imprime “fechando conta normalí. 

8. A resposta certa é (a). O código não compila pois, ao sobrescrevê-lo, 
tentamos definir um escopo menor. Não compila também porque o 
método fecha é package protected dentro de Conta. 

9. A resposta certa é (c). O código compila e imprime “fechando finan- 


ceiro. 
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10. O código compila normalmente. Apesar de o método ligar não es- 


tar declarado na classe CarroConcreto, a classe herda este método 
de Carro, logo, não é necessário reescrevê-lo (poderia reescrever se 
achasse necessário). 


A declaração de que CarroConcreto implementa Veiculo tam- 
bém não era necessária, pois Carro já implementa Veiculo e 


CarroConcreto éum Carro. 


9.4 - Exercícios 
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1. A resposta certa é (c). O código compila e roda, ao rodar não dá excep- 


tion. 


. A resposta certa é (d). O código compila e roda, ao rodar dá exception. 


. A resposta certa é (a). O código não compila: D até implementa ze w 


mas não implementa Y. 


. A resposta certa é (d). O código compila: algum subtipo de D pode 


implementar Y. Ao rodar, ele dá exception. 


. A resposta certa é (c). Compila, pois apesar de B não implementar z, 


um subtipo dele pode (e na prática já o faz) implementá-lo. Ao rodar 
não dá exception nenhuma. 


. A resposta certa é (d). O código compila e roda dando exception. 
. A resposta certa é (a). D não implementa Y, não compila. 


. A resposta certa é (b). O código não compila: instanceof é minús- 


culo. 


. A resposta certa é (c). O código compila e imprime true. 
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9.5 - 


1 


10 


Exercícios 


. A resposta certa é (c). O código compila e imprime 2. 
A resposta certa é (a). O código não compila. 
A resposta certa é (a). O código não compila. 


A resposta certa é (a). O código não compila: não faz sentido acessar o 
} super de outro objeto que não eu mesmo. 


. A resposta certa é (c). O código compila e não imprime nada: o método 


definido não é um construtor! 

A resposta certa é (a). Não compila pois tentamos invocar dois this. 
A resposta certa é (c). Compila e não imprime nada. 

. À resposta certa é (c). Compila e não imprime nada. 


. A resposta certa é (a). Não compila: não podemos referenciar um 


método de instância ao invocar um construtor this. 


. A resposta certa é (b). Compila e imprime “2. 


9.6 - Exercícios 


1 


. A resposta certa é (c). Compila e imprime “2º, 

A resposta certa é (a). A classe B não compila. 

. A resposta certa é (b). Compila e imprime “1. 

A resposta certa é (b). Compila e imprime “1. 

. A resposta certa é (a). Não compila, pois o método é final. 


A resposta certa é (c). Compila e imprime ‘b. 


389 


Casa do Código 





10.1 - Exercícios 


1. A resposta certa é (d). A única exception da lista que não é checked é a 





IndexOutOfBoundsException 





10.2 - Exercícios 


1. (b) e (e) são corretas. (a) está errada pois podemos usar exceptions 


mesmo sem entradas do usuário. (c) está errada pois podemos manter 
o programa rodando mesmo que uma exception ocorra. (d) está errada 
pois devemos usar outras estruturas para controlar o fluxo de nosso 


programa, como if por exemplo. 


2. (a) e (c) estão corretas. (b) e (d) estão incorretas por serem os opos- 


tos das certas, e (e) está incorreta pois exceptions não são maneiras de 


aumentar a segurança. 


10.3 - Exercícios 


1. A resposta certa é (a). O código não compila pois a variável local nunca 


foi inicializada. 


2. À resposta certa é (c). Quando ocorre a exception, o fluxo desvia para 


«o» 


imprimir “b” e depois continua normal com o “c”. 


3. A resposta certa é (c). Quando ocorre a exception, o fluxo desvia para 


imprimir “b”, passa pelo finally imprimindo “c” e depois continua 
normal com o “d”. 


10.4 - Exercícios 


1. A resposta certa é (a). Devemos colocar uma 





java.io.IOException. 


2. A resposta certa é (b). O código compila pois ele cria um array de di- 
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mensão 2. Ele imprime acefdb. 


Casa do Código Capítulo 12. Respostas dos Exercícios 





3. A resposta certa é (c). O código compila e imprime ace, jogando uma 





Exception. 


4. À resposta certa é (a). O código não compila pois o método m2 deve 





tratar ou jogar java.io.FileNotFoundException. 


5. A resposta certa é (a). O código não compila pois o método main deve 





tratar ou jogar java.io.IOException. 


6. A resposta certa é (c). Compila, imprime ace ejogaa Excepion. 





7. A resposta certa é (e). Compila, e imprime acedb. 


8. A resposta certa é (b). Compila, e imprime acefdb, note que não 
jogamos a exception, somente a instanciamos. 


9. A resposta certa é (a). Não compila: o sSystem.out do f é 


unreachable. 


10. A resposta certa é (a). Não compila: a palavra throw deveria ter sido 
usada para jogar a Exception. 


11. A resposta certa é (c). Compila, e imprime ace e estoura uma 





Exception. 


10.5 - Exercícios 





1. A resposta certa é (e), StackOverflowError. 





2. A resposta certa é (d). OutOfMemoryError, pois tem um loop in- 
finito. 
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